diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..60839c7
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,2 @@
+_saves/
+_server/config.json
\ No newline at end of file
diff --git a/_saves/.exported b/_saves/.exported
deleted file mode 100644
index 56a6051..0000000
--- a/_saves/.exported
+++ /dev/null
@@ -1 +0,0 @@
-1
\ No newline at end of file
diff --git a/_saves/Eustia_autoSave b/_saves/Eustia_autoSave
deleted file mode 100644
index 2a3d507..0000000
--- a/_saves/Eustia_autoSave
+++ /dev/null
@@ -1 +0,0 @@
-N4IgZgNg9lBOCSATEAuEBnALrApjzADAIwgA0IAFjrFKqAJYC2AhgOY6qXVQB0A7jgBGABzIhmAOybNMHFGGYR0OchOaM5IAMLMmYiADdURchWEsAHqgCct66dEoiBF+RZrLqALQmQ75qgE5DIA1sauIIg4YOFBflExKHHownjISW5QEjgAnoHkOBaOcTgAjgCu9OY4EpioANoAuuT0sozodCAAxllYkpgdKKCCMGFO5JB5TgC+5JgwSnSzIGWVwoOg6HxwiCQzy9BdnYj0uF2Y9Fmc5aLkVigArORTzsuQbBsgrTiMACL4uggqGw5RUIHKygAQvgBDUADL0VgUOooEFggD66OUOEQmOMAHYAGwEawADgATOSACxxTGwSS49EE4lkyk04K4ZiDeqgDyaAA65SpOAIgkFDy6VPJYhY6wa2l0jAoUA0IEas156gFQpFYvKEqlMuYcpQ9RAACt6OVWOVJGqNSA+ZxBcLReLJdK3MbuSAANIAFQAavbSJrVWhBfjyYTyZHyWASF6TWaA4GAGLS9Whx1a53lKMxuMJo3JkCsKAQRAhXIhsPa116g2evze+WCegUCHlWhZut513Md2GpM+y3MKCYSSsWs58MgF0iwf6j0ln3oKgSVgSAzK1W92f1xdD5uytcUejb3ccfdOiP56Ox+/Fkfyy048cz2/z++Fp+JlullguD4MQYhAXghDNjaU5dBQU5iNBm6wVOzYWPQ45oVOoF3OhUCYZuBCobh+GsAAzJ+uYoBI5QQBAL6miAOT0BUJA3pR1G0fRZpMRUmYOreHF0QBPo8eU5FseGglcYxzFChRkk0UJp7yqJDzyXIUnCSpsmEupqCacpDGifielUYp0miaSpkGa2RmydY1nmVpdksQQ9rNBgIRVKgChKGCcGwOktLogFiBaBWcB4mguDIOQmLfIwAASF6YFFTRxeiEA4MwBg4HCUBdFFwDLJiBj0Og3yMp04EgXsaLLIIMJ4BIACCmCTl0IS/DIARmZxICYswOD0GlHmYic6DMIIWVVUMGDYBBoGmvURCkQ8zRNOqbwVtAAiwNyHlYDgJq+H0FxYPQXSfPMk4QP6TByOSDxEEQUpRjwhJ2C91hPN05SwLA93hkQDxUtYPCkXYUNQ4SDj5CAggyJgWXw4wWQ1hkKxFPDiPtVl3UsOw8PCFA5VZATbByCUFjYMwFNE5jaO5b8pw4OcEBTHEiISHAOAAMqyCaySTrAKKElSDykTwVLTMsoWcKu1ULbVnSyg0QQa6Qmva1ruv1D4pAG0bJgG0ExuGyblsWxbzQa1t5A0OUsicL6UAZloVKsAAvGIBiKKCGzLLl+2XBIebWDgpGIOKggPAQkbMNY0flNYhHx/qpKg+6BBgIK1iEkQMTkDa9DpCA5LMIghKEog1iFYgYBgFS6JUqSifooIVKEmA6JgIgU0EDgNJV3gYgXMD+KkficfWKRBBSi40xAA==
\ No newline at end of file
diff --git a/_server/CodeMirror/defs.js b/_server/CodeMirror/defs.js
index 1c16b25..ee517af 100644
--- a/_server/CodeMirror/defs.js
+++ b/_server/CodeMirror/defs.js
@@ -1,4130 +1,4526 @@
 var terndefs_f6783a0a_522d_417e_8407_94c67b692e50 = [
   {
     "!name": "browser",
-    "Node": {
+    Node: {
       "!type": "fn()",
-      "prototype": {
-        "nextSibling": {
+      prototype: {
+        nextSibling: {
           "!type": "+Element",
-          "!doc": "返回紧接其父节点的childNodes列表中指定节点之后的节点;如果指定节点是该列表中的最后一个节点,则返回null.",
+          "!doc":
+            "返回紧接其父节点的childNodes列表中指定节点之后的节点;如果指定节点是该列表中的最后一个节点,则返回null.",
         },
-        "previousSibling": {
+        previousSibling: {
           "!type": "+Element",
-          "!doc": "返回紧接其父节点的childNodes列表中指定节点之前的节点,如果指定节点是该列表中的第一个节点,则返回null.",
+          "!doc":
+            "返回紧接其父节点的childNodes列表中指定节点之前的节点,如果指定节点是该列表中的第一个节点,则返回null.",
         },
-        "lastChild": {
+        lastChild: {
           "!type": "+Element",
-          "!doc": "返回节点的最后一个孩子."
+          "!doc": "返回节点的最后一个孩子.",
         },
-        "firstChild": {
+        firstChild: {
           "!type": "+Element",
-          "!doc": "返回树中该节点的第一个子节点;如果该节点为无子节点,则返回null.如果该节点是Document,则返回其直接子节点列表中的第一个节点.",
+          "!doc":
+            "返回树中该节点的第一个子节点;如果该节点为无子节点,则返回null.如果该节点是Document,则返回其直接子节点列表中的第一个节点.",
         },
-        "childNodes": {
+        childNodes: {
           "!type": "+NodeList",
-          "!doc": "返回给定元素的子节点的集合."
+          "!doc": "返回给定元素的子节点的集合.",
         },
-        "parentNode": {
+        parentNode: {
           "!type": "+Element",
-          "!doc": "返回DOM树中指定节点的父级."
+          "!doc": "返回DOM树中指定节点的父级.",
         },
-        "tagName": {
+        tagName: {
           "!type": "string",
-          "!doc": "将当前节点的名称作为字符串返回."
+          "!doc": "将当前节点的名称作为字符串返回.",
         },
-        "insertBefore": {
+        insertBefore: {
           "!type": "fn(newElt: +Element, before: +Element) -> +Element",
           "!doc": "将指定的节点插入到参考元素之前,作为当前节点的子级.",
         },
-        "removeChild": {
+        removeChild: {
           "!type": "fn(oldNode: +Element) -> +Element",
           "!doc": "从DOM中删除一个子节点.返回已删除的节点.",
         },
-        "appendChild": {
+        appendChild: {
           "!type": "fn(newNode: +Element) -> +Element",
-          "!doc": "将一个节点添加到指定父节点的子节点列表的末尾.如果该节点已经存在,则将其从当前父节点中删除,然后添加到新的父节点中.",
+          "!doc":
+            "将一个节点添加到指定父节点的子节点列表的末尾.如果该节点已经存在,则将其从当前父节点中删除,然后添加到新的父节点中.",
         },
-        "cloneNode": {
+        cloneNode: {
           "!type": "fn(deep: bool) -> +Element",
-          "!doc": "返回在其上调用此方法的节点的副本."
+          "!doc": "返回在其上调用此方法的节点的副本.",
         },
-        "addEventListener": {
+        addEventListener: {
           "!type": "fn(type: string, listener: fn(e: +Event), capture: bool)",
-          "!doc": "在单个目标上注册单个事件侦听器.事件目标可以是文档中的单个元素,文档本身,窗口或XMLHttpRequest.",
+          "!doc":
+            "在单个目标上注册单个事件侦听器.事件目标可以是文档中的单个元素,文档本身,窗口或XMLHttpRequest.",
         },
-        "removeEventListener": {
+        removeEventListener: {
           "!type": "fn(type: string, listener: fn(), capture: bool)",
           "!doc": "允许从事件目标中删除事件侦听器.",
         },
-        "innerText": {
+        innerText: {
           "!type": "string",
-          "!doc": "获取或设置节点及其后代的文本内容."
-        }
-      },
-      "!doc": "节点是一个接口,许多DOM类型都从该接口继承,并允许类似地对待(或测试)这些各种类型.",
-    },
-    "Element": {
-      "!type": "fn()",
-      "prototype": {
-        "!proto": "Node.prototype",
-        "getAttribute": {
-          "!type": "fn(name: string) -> string",
-          "!doc": "返回指定元素上的命名属性的值.如果命名属性不存在,则返回的值将为null或\" \"(空字符串).",
+          "!doc": "获取或设置节点及其后代的文本内容.",
         },
-        "setAttribute": {
+      },
+      "!doc":
+        "节点是一个接口,许多DOM类型都从该接口继承,并允许类似地对待(或测试)这些各种类型.",
+    },
+    Element: {
+      "!type": "fn()",
+      prototype: {
+        "!proto": "Node.prototype",
+        getAttribute: {
+          "!type": "fn(name: string) -> string",
+          "!doc":
+            '返回指定元素上的命名属性的值.如果命名属性不存在,则返回的值将为null或" "(空字符串).',
+        },
+        setAttribute: {
           "!type": "fn(name: string, value: string)",
           "!doc": "在指定元素上添加新属性或更改现有属性的值.",
         },
-        "removeAttribute": {
+        removeAttribute: {
           "!type": "fn(name: string)",
           "!doc": "从指定元素中删除属性.",
         },
-        "getElementsByTagName": {
+        getElementsByTagName: {
           "!type": "fn(tagName: string) -> +NodeList",
-          "!doc": "返回具有给定标签名的元素列表.搜索指定元素下面的子树,不包括元素本身.返回的列表是活动的,这意味着它将自动使用DOM树进行更新.因此,无需使用相同的元素和参数多次调用element.getElementsByTagName."
+          "!doc":
+            "返回具有给定标签名的元素列表.搜索指定元素下面的子树,不包括元素本身.返回的列表是活动的,这意味着它将自动使用DOM树进行更新.因此,无需使用相同的元素和参数多次调用element.getElementsByTagName.",
         },
-        "getElementsByClassName": {
+        getElementsByClassName: {
           "!type": "fn(name: string) -> +NodeList",
-          "!doc": "返回具有所有给定类名称的一组元素.在文档对象上调用时,将搜索整个文档,包括根节点.您还可以在任何元素上调用getElementsByClassName;它将仅返回元素,它们是具有给定类名的指定根元素的后代."
+          "!doc":
+            "返回具有所有给定类名称的一组元素.在文档对象上调用时,将搜索整个文档,包括根节点.您还可以在任何元素上调用getElementsByClassName;它将仅返回元素,它们是具有给定类名的指定根元素的后代.",
         },
-        "children": {
+        children: {
           "!type": "+HTMLCollection",
-          "!doc": "返回给定元素的子元素的集合."
+          "!doc": "返回给定元素的子元素的集合.",
         },
-        "className": {
+        className: {
           "!type": "string",
           "!doc": "获取并设置指定元素的class属性的值.",
         },
-        "style": {
-          "cssText": "string",
-          "alignmentBaseline": "string",
-          "background": "string",
-          "backgroundAttachment": "string",
-          "backgroundClip": "string",
-          "backgroundColor": "string",
-          "backgroundImage": "string",
-          "backgroundOrigin": "string",
-          "backgroundPosition": "string",
-          "backgroundPositionX": "string",
-          "backgroundPositionY": "string",
-          "backgroundRepeat": "string",
-          "backgroundRepeatX": "string",
-          "backgroundRepeatY": "string",
-          "backgroundSize": "string",
-          "baselineShift": "string",
-          "border": "string",
-          "borderBottom": "string",
-          "borderBottomColor": "string",
-          "borderBottomLeftRadius": "string",
-          "borderBottomRightRadius": "string",
-          "borderBottomStyle": "string",
-          "borderBottomWidth": "string",
-          "borderCollapse": "string",
-          "borderColor": "string",
-          "borderImage": "string",
-          "borderImageOutset": "string",
-          "borderImageRepeat": "string",
-          "borderImageSlice": "string",
-          "borderImageSource": "string",
-          "borderImageWidth": "string",
-          "borderLeft": "string",
-          "borderLeftColor": "string",
-          "borderLeftStyle": "string",
-          "borderLeftWidth": "string",
-          "borderRadius": "string",
-          "borderRight": "string",
-          "borderRightColor": "string",
-          "borderRightStyle": "string",
-          "borderRightWidth": "string",
-          "borderSpacing": "string",
-          "borderStyle": "string",
-          "borderTop": "string",
-          "borderTopColor": "string",
-          "borderTopLeftRadius": "string",
-          "borderTopRightRadius": "string",
-          "borderTopStyle": "string",
-          "borderTopWidth": "string",
-          "borderWidth": "string",
-          "bottom": "string",
-          "boxShadow": "string",
-          "boxSizing": "string",
-          "captionSide": "string",
-          "clear": "string",
-          "clip": "string",
-          "clipPath": "string",
-          "clipRule": "string",
-          "color": "string",
-          "colorInterpolation": "string",
-          "colorInterpolationFilters": "string",
-          "colorProfile": "string",
-          "colorRendering": "string",
-          "content": "string",
-          "counterIncrement": "string",
-          "counterReset": "string",
-          "cursor": "string",
-          "direction": "string",
-          "display": "string",
-          "dominantBaseline": "string",
-          "emptyCells": "string",
-          "enableBackground": "string",
-          "fill": "string",
-          "fillOpacity": "string",
-          "fillRule": "string",
-          "filter": "string",
-          "float": "string",
-          "floodColor": "string",
-          "floodOpacity": "string",
-          "font": "string",
-          "fontFamily": "string",
-          "fontSize": "string",
-          "fontStretch": "string",
-          "fontStyle": "string",
-          "fontVariant": "string",
-          "fontWeight": "string",
-          "glyphOrientationHorizontal": "string",
-          "glyphOrientationVertical": "string",
-          "height": "string",
-          "imageRendering": "string",
-          "kerning": "string",
-          "left": "string",
-          "letterSpacing": "string",
-          "lightingColor": "string",
-          "lineHeight": "string",
-          "listStyle": "string",
-          "listStyleImage": "string",
-          "listStylePosition": "string",
-          "listStyleType": "string",
-          "margin": "string",
-          "marginBottom": "string",
-          "marginLeft": "string",
-          "marginRight": "string",
-          "marginTop": "string",
-          "marker": "string",
-          "markerEnd": "string",
-          "markerMid": "string",
-          "markerStart": "string",
-          "mask": "string",
-          "maxHeight": "string",
-          "maxWidth": "string",
-          "minHeight": "string",
-          "minWidth": "string",
-          "opacity": "string",
-          "orphans": "string",
-          "outline": "string",
-          "outlineColor": "string",
-          "outlineOffset": "string",
-          "outlineStyle": "string",
-          "outlineWidth": "string",
-          "overflow": "string",
-          "overflowWrap": "string",
-          "overflowX": "string",
-          "overflowY": "string",
-          "padding": "string",
-          "paddingBottom": "string",
-          "paddingLeft": "string",
-          "paddingRight": "string",
-          "paddingTop": "string",
-          "page": "string",
-          "pageBreakAfter": "string",
-          "pageBreakBefore": "string",
-          "pageBreakInside": "string",
-          "pointerEvents": "string",
-          "position": "string",
-          "quotes": "string",
-          "resize": "string",
-          "right": "string",
-          "shapeRendering": "string",
-          "size": "string",
-          "speak": "string",
-          "src": "string",
-          "stopColor": "string",
-          "stopOpacity": "string",
-          "stroke": "string",
-          "strokeDasharray": "string",
-          "strokeDashoffset": "string",
-          "strokeLinecap": "string",
-          "strokeLinejoin": "string",
-          "strokeMiterlimit": "string",
-          "strokeOpacity": "string",
-          "strokeWidth": "string",
-          "tabSize": "string",
-          "tableLayout": "string",
-          "textAlign": "string",
-          "textAnchor": "string",
-          "textDecoration": "string",
-          "textIndent": "string",
-          "textLineThrough": "string",
-          "textLineThroughColor": "string",
-          "textLineThroughMode": "string",
-          "textLineThroughStyle": "string",
-          "textLineThroughWidth": "string",
-          "textOverflow": "string",
-          "textOverline": "string",
-          "textOverlineColor": "string",
-          "textOverlineMode": "string",
-          "textOverlineStyle": "string",
-          "textOverlineWidth": "string",
-          "textRendering": "string",
-          "textShadow": "string",
-          "textTransform": "string",
-          "textUnderline": "string",
-          "textUnderlineColor": "string",
-          "textUnderlineMode": "string",
-          "textUnderlineStyle": "string",
-          "textUnderlineWidth": "string",
-          "top": "string",
-          "unicodeBidi": "string",
-          "unicodeRange": "string",
-          "vectorEffect": "string",
-          "verticalAlign": "string",
-          "visibility": "string",
-          "whiteSpace": "string",
-          "width": "string",
-          "wordBreak": "string",
-          "wordSpacing": "string",
-          "wordWrap": "string",
-          "writingMode": "string",
-          "zIndex": "string",
-          "zoom": "string",
-          "!doc": "返回一个表示元素的style属性的对象."
+        style: {
+          cssText: "string",
+          alignmentBaseline: "string",
+          background: "string",
+          backgroundAttachment: "string",
+          backgroundClip: "string",
+          backgroundColor: "string",
+          backgroundImage: "string",
+          backgroundOrigin: "string",
+          backgroundPosition: "string",
+          backgroundPositionX: "string",
+          backgroundPositionY: "string",
+          backgroundRepeat: "string",
+          backgroundRepeatX: "string",
+          backgroundRepeatY: "string",
+          backgroundSize: "string",
+          baselineShift: "string",
+          border: "string",
+          borderBottom: "string",
+          borderBottomColor: "string",
+          borderBottomLeftRadius: "string",
+          borderBottomRightRadius: "string",
+          borderBottomStyle: "string",
+          borderBottomWidth: "string",
+          borderCollapse: "string",
+          borderColor: "string",
+          borderImage: "string",
+          borderImageOutset: "string",
+          borderImageRepeat: "string",
+          borderImageSlice: "string",
+          borderImageSource: "string",
+          borderImageWidth: "string",
+          borderLeft: "string",
+          borderLeftColor: "string",
+          borderLeftStyle: "string",
+          borderLeftWidth: "string",
+          borderRadius: "string",
+          borderRight: "string",
+          borderRightColor: "string",
+          borderRightStyle: "string",
+          borderRightWidth: "string",
+          borderSpacing: "string",
+          borderStyle: "string",
+          borderTop: "string",
+          borderTopColor: "string",
+          borderTopLeftRadius: "string",
+          borderTopRightRadius: "string",
+          borderTopStyle: "string",
+          borderTopWidth: "string",
+          borderWidth: "string",
+          bottom: "string",
+          boxShadow: "string",
+          boxSizing: "string",
+          captionSide: "string",
+          clear: "string",
+          clip: "string",
+          clipPath: "string",
+          clipRule: "string",
+          color: "string",
+          colorInterpolation: "string",
+          colorInterpolationFilters: "string",
+          colorProfile: "string",
+          colorRendering: "string",
+          content: "string",
+          counterIncrement: "string",
+          counterReset: "string",
+          cursor: "string",
+          direction: "string",
+          display: "string",
+          dominantBaseline: "string",
+          emptyCells: "string",
+          enableBackground: "string",
+          fill: "string",
+          fillOpacity: "string",
+          fillRule: "string",
+          filter: "string",
+          float: "string",
+          floodColor: "string",
+          floodOpacity: "string",
+          font: "string",
+          fontFamily: "string",
+          fontSize: "string",
+          fontStretch: "string",
+          fontStyle: "string",
+          fontVariant: "string",
+          fontWeight: "string",
+          glyphOrientationHorizontal: "string",
+          glyphOrientationVertical: "string",
+          height: "string",
+          imageRendering: "string",
+          kerning: "string",
+          left: "string",
+          letterSpacing: "string",
+          lightingColor: "string",
+          lineHeight: "string",
+          listStyle: "string",
+          listStyleImage: "string",
+          listStylePosition: "string",
+          listStyleType: "string",
+          margin: "string",
+          marginBottom: "string",
+          marginLeft: "string",
+          marginRight: "string",
+          marginTop: "string",
+          marker: "string",
+          markerEnd: "string",
+          markerMid: "string",
+          markerStart: "string",
+          mask: "string",
+          maxHeight: "string",
+          maxWidth: "string",
+          minHeight: "string",
+          minWidth: "string",
+          opacity: "string",
+          orphans: "string",
+          outline: "string",
+          outlineColor: "string",
+          outlineOffset: "string",
+          outlineStyle: "string",
+          outlineWidth: "string",
+          overflow: "string",
+          overflowWrap: "string",
+          overflowX: "string",
+          overflowY: "string",
+          padding: "string",
+          paddingBottom: "string",
+          paddingLeft: "string",
+          paddingRight: "string",
+          paddingTop: "string",
+          page: "string",
+          pageBreakAfter: "string",
+          pageBreakBefore: "string",
+          pageBreakInside: "string",
+          pointerEvents: "string",
+          position: "string",
+          quotes: "string",
+          resize: "string",
+          right: "string",
+          shapeRendering: "string",
+          size: "string",
+          speak: "string",
+          src: "string",
+          stopColor: "string",
+          stopOpacity: "string",
+          stroke: "string",
+          strokeDasharray: "string",
+          strokeDashoffset: "string",
+          strokeLinecap: "string",
+          strokeLinejoin: "string",
+          strokeMiterlimit: "string",
+          strokeOpacity: "string",
+          strokeWidth: "string",
+          tabSize: "string",
+          tableLayout: "string",
+          textAlign: "string",
+          textAnchor: "string",
+          textDecoration: "string",
+          textIndent: "string",
+          textLineThrough: "string",
+          textLineThroughColor: "string",
+          textLineThroughMode: "string",
+          textLineThroughStyle: "string",
+          textLineThroughWidth: "string",
+          textOverflow: "string",
+          textOverline: "string",
+          textOverlineColor: "string",
+          textOverlineMode: "string",
+          textOverlineStyle: "string",
+          textOverlineWidth: "string",
+          textRendering: "string",
+          textShadow: "string",
+          textTransform: "string",
+          textUnderline: "string",
+          textUnderlineColor: "string",
+          textUnderlineMode: "string",
+          textUnderlineStyle: "string",
+          textUnderlineWidth: "string",
+          top: "string",
+          unicodeBidi: "string",
+          unicodeRange: "string",
+          vectorEffect: "string",
+          verticalAlign: "string",
+          visibility: "string",
+          whiteSpace: "string",
+          width: "string",
+          wordBreak: "string",
+          wordSpacing: "string",
+          wordWrap: "string",
+          writingMode: "string",
+          zIndex: "string",
+          zoom: "string",
+          "!doc": "返回一个表示元素的style属性的对象.",
         },
-        "classList": {
+        classList: {
           "!type": "+DOMTokenList",
-          "!doc": "返回元素的class属性的标记列表."
+          "!doc": "返回元素的class属性的标记列表.",
         },
-        "title": {
+        title: {
           "!type": "string",
-          "!doc": " \"\u5efa\u7acb\u5f53\u9f20\u6807\u60ac\u505c\u5728\u663e\u793a\u7684\u8282\u70b9\u4e0a\u65f6\u5728\"\u5de5\u5177\u63d0\u793a\"\u5f39\u51fa\u7a97\u53e3\u4e2d\u663e\u793a\u7684\u6587\u672c.\","
+          "!doc":
+            ' "\u5efa\u7acb\u5f53\u9f20\u6807\u60ac\u505c\u5728\u663e\u793a\u7684\u8282\u70b9\u4e0a\u65f6\u5728"\u5de5\u5177\u63d0\u793a"\u5f39\u51fa\u7a97\u53e3\u4e2d\u663e\u793a\u7684\u6587\u672c.",',
         },
-        "width": {
+        width: {
           "!type": "number",
-          "!doc": "返回元素的布局宽度."
+          "!doc": "返回元素的布局宽度.",
         },
-        "height": {
+        height: {
           "!type": "number",
-          "!doc": "元素相对于元素的offsetParent的高度."
+          "!doc": "元素相对于元素的offsetParent的高度.",
         },
-        "getContext": {
+        getContext: {
           "!type": "fn(id: string) -> CanvasRenderingContext2D",
-          "!doc": " DOM画布元素公开了HTMLCanvasElement接口,该接口提供了用于操纵画布元素的布局和表示的属性和方法.HTMLCanvasElement接口继承了元素对象接口的属性和方法.",
+          "!doc":
+            " DOM画布元素公开了HTMLCanvasElement接口,该接口提供了用于操纵画布元素的布局和表示的属性和方法.HTMLCanvasElement接口继承了元素对象接口的属性和方法.",
         },
-        "innerHTML": {
+        innerHTML: {
           "!type": "string",
           "!doc": "设置或获取描述元素后代的HTML语法.",
-        }
+        },
       },
       "!doc": "表示HTML或XML文档中的元素.",
     },
-    "Document": {
+    Document: {
       "!type": "fn()",
-      "prototype": {
+      prototype: {
         "!proto": "Node.prototype",
-        "height": {
+        height: {
           "!type": "number",
           "!doc": "返回当前文档的<body>元素的高度.",
         },
-        "width": {
+        width: {
           "!type": "number",
           "!doc": "以像素为单位返回当前文档的<body>元素的宽度.",
         },
-        "body": {
+        body: {
           "!type": "+Element",
           "!doc": "返回当前文档的<body>或<frameset>节点.",
         },
-        "cookie": {
+        cookie: {
           "!type": "string",
           "!doc": "获取并设置与当前文档关联的cookie.",
         },
-        "URL": "string",
-        "title": {
+        URL: "string",
+        title: {
           "!type": "string",
-          "!doc": "获取或设置文档的标题."
+          "!doc": "获取或设置文档的标题.",
         },
-        "getElementById": {
+        getElementById: {
           "!type": "fn(id: string) -> +Element",
-          "!doc": "通过元素ID返回对该元素的引用."
+          "!doc": "通过元素ID返回对该元素的引用.",
         },
-        "getElementsByTagName": {
+        getElementsByTagName: {
           "!type": "fn(tagName: string) -> +NodeList",
-          "!doc": "返回具有给定标签名称的元素的NodeList.将搜索整个文档,包括根节点.返回的NodeList处于活动状态,这意味着它会自动更新自身以与DOM树保持同步,而无需再次调用document.getElementsByTagName."
+          "!doc":
+            "返回具有给定标签名称的元素的NodeList.将搜索整个文档,包括根节点.返回的NodeList处于活动状态,这意味着它会自动更新自身以与DOM树保持同步,而无需再次调用document.getElementsByTagName.",
         },
-        "getElementsByName": {
+        getElementsByName: {
           "!type": "fn(name: string) -> +HTMLCollection",
           "!doc": "返回HTML文档中具有给定名称的元素列表.",
         },
-        "getElementsByClassName": "Element.prototype.getElementsByClassName"
+        getElementsByClassName: "Element.prototype.getElementsByClassName",
       },
-      "!doc": "浏览器中加载的每个网页都有其自己的文档对象.此对象用作网页内容(DOM树,包括诸如<body>和<table>之类的元素)的入口点,并提供文档的全局功能(例如获取页面的URL和在文档中创建新元素)."
+      "!doc":
+        "浏览器中加载的每个网页都有其自己的文档对象.此对象用作网页内容(DOM树,包括诸如<body>和<table>之类的元素)的入口点,并提供文档的全局功能(例如获取页面的URL和在文档中创建新元素).",
     },
-    "document": {
+    document: {
       "!type": "+Document",
-      "!doc": "浏览器中加载的每个网页都有其自己的文档对象.此对象用作网页内容(DOM树,包括诸如<body>和<table>之类的元素)的入口点,并提供文档的全局功能(例如获取页面的URL和在文档中创建新元素)."
+      "!doc":
+        "浏览器中加载的每个网页都有其自己的文档对象.此对象用作网页内容(DOM树,包括诸如<body>和<table>之类的元素)的入口点,并提供文档的全局功能(例如获取页面的URL和在文档中创建新元素).",
     },
-    "Event": {
+    Event: {
       "!type": "fn()",
-      "prototype": {
-        "stopPropagation": {
+      prototype: {
+        stopPropagation: {
           "!type": "fn()",
-          "!doc": "防止当前事件进一步传播."
+          "!doc": "防止当前事件进一步传播.",
         },
-        "preventDefault": {
+        preventDefault: {
           "!type": "fn()",
-          "!doc": "如果可以取消事件,则取消该事件,而不停止事件的进一步传播."
+          "!doc": "如果可以取消事件,则取消该事件,而不停止事件的进一步传播.",
         },
-        "stopImmediatePropagation": {
+        stopImmediatePropagation: {
           "!type": "fn()",
-          "!doc": "防止同一事件的其他侦听器被调用."
+          "!doc": "防止同一事件的其他侦听器被调用.",
         },
-        "type": {
+        type: {
           "!type": "string",
-          "!doc": "返回包含事件类型的字符串."
+          "!doc": "返回包含事件类型的字符串.",
         },
-        "target": {
+        target: {
           "!type": "+Element",
-          "!doc": " EventTarget是由对象实现的DOM接口,这些对象可以接收DOM事件并具有侦听器.最常见的EventTarget是DOM元素,尽管其他对象也可以是EventTarget,例如文档,窗口,XMLHttpRequest,和别的."
+          "!doc":
+            " EventTarget是由对象实现的DOM接口,这些对象可以接收DOM事件并具有侦听器.最常见的EventTarget是DOM元素,尽管其他对象也可以是EventTarget,例如文档,窗口,XMLHttpRequest,和别的.",
         },
-        "clientX": {
+        clientX: {
           "!type": "number",
-          "!doc": "返回事件发生的应用程序客户区域内的水平坐标(与页面内的坐标相反).例如,单击客户区域左上角将始终显示clientX值为0的鼠标事件,无论页面是否水平滚动."
+          "!doc":
+            "返回事件发生的应用程序客户区域内的水平坐标(与页面内的坐标相反).例如,单击客户区域左上角将始终显示clientX值为0的鼠标事件,无论页面是否水平滚动.",
         },
-        "clientY": {
+        clientY: {
           "!type": "number",
-          "!doc": "返回事件发生在应用程序客户区中的垂直坐标(与页面中的坐标相反).例如,单击客户区左上角将始终显示不管页面是否垂直滚动,clientY值为0的鼠标事件."
+          "!doc":
+            "返回事件发生在应用程序客户区中的垂直坐标(与页面中的坐标相反).例如,单击客户区左上角将始终显示不管页面是否垂直滚动,clientY值为0的鼠标事件.",
         },
-        "keyCode": {
+        keyCode: {
           "!type": "number",
-          "!doc": "返回按键事件中的非字符键或任何其他类型的键盘事件中的任何键的Unicode值.",
+          "!doc":
+            "返回按键事件中的非字符键或任何其他类型的键盘事件中的任何键的Unicode值.",
         },
-        "charCode": {
+        charCode: {
           "!type": "number",
-          "!doc": "返回在按键事件期间按下的字符键的Unicode值."
+          "!doc": "返回在按键事件期间按下的字符键的Unicode值.",
         },
-        "which": {
+        which: {
           "!type": "number",
-          "!doc": "返回所按下键的数字keyCode或所按下字母数字键的字符代码(charCode)."
+          "!doc":
+            "返回所按下键的数字keyCode或所按下字母数字键的字符代码(charCode).",
         },
-        "button": {
+        button: {
           "!type": "number",
-          "!doc": "指示导致事件的鼠标按钮."
+          "!doc": "指示导致事件的鼠标按钮.",
         },
-        "shiftKey": {
+        shiftKey: {
           "!type": "bool",
           "!doc": "指示事件触发时是否按下SHIFT键.",
         },
-        "ctrlKey": {
+        ctrlKey: {
           "!type": "bool",
           "!doc": "指示事件触发时是否按下了CTRL键.",
         },
-        "altKey": {
+        altKey: {
           "!type": "bool",
           "!doc": "指示事件触发时是否按下ALT键.",
-        }
-      }
+        },
+      },
     },
-    "Storage": {
-      "length": {
+    Storage: {
+      length: {
         "!type": "number",
-        "!doc": "存储接口的length只读属性返回一个整数,该整数表示存储在存储对象中的数据项的数量.",
+        "!doc":
+          "存储接口的length只读属性返回一个整数,该整数表示存储在存储对象中的数据项的数量.",
       },
-      "setItem": {
+      setItem: {
         "!type": "fn(name: string, value: string)",
-        "!doc": "存储接口的setItem()方法在传递键名称和值时,会将该键添加到存储中,或者更新该键的值(如果已存在).",
+        "!doc":
+          "存储接口的setItem()方法在传递键名称和值时,会将该键添加到存储中,或者更新该键的值(如果已存在).",
       },
-      "getItem": {
+      getItem: {
         "!type": "fn(name: string) -> string",
         "!doc": "存储接口的getItem()方法在传递键名时将返回该键的值.",
       },
-      "key": {
+      key: {
         "!type": "fn(index: number) -> string",
-        "!doc": "存储接口的key()方法传递数字n时,返回存储中第n个键的名称.键的顺序是用户代理定义的,因此您不应依赖它."
+        "!doc":
+          "存储接口的key()方法传递数字n时,返回存储中第n个键的名称.键的顺序是用户代理定义的,因此您不应依赖它.",
       },
-      "removeItem": {
+      removeItem: {
         "!type": "fn(key: string)",
         "!doc": "存储接口的removeItem()方法在传递了键名后,将从存储中删除该键.",
       },
-      "clear": {
+      clear: {
         "!type": "fn()",
-        "!doc": "存储接口的clear()方法在被调用时将从存储中清空所有键."
-      }
+        "!doc": "存储接口的clear()方法在被调用时将从存储中清空所有键.",
+      },
     },
-    "localStorage": {
+    localStorage: {
       "!type": "Storage",
-      "!doc": " localStorage属性允许您访问本地存储对象.localStorage与sessionStorage类似.唯一的区别是,虽然存储在localStorage中的数据没有到期时间,但是浏览会话时存储在sessionStorage中的数据将被清除.结束-也就是说,当浏览器关闭时.\ n \ n请注意,存储在localStorage或sessionStorage中的数据特定于页面协议."
+      "!doc":
+        " localStorage属性允许您访问本地存储对象.localStorage与sessionStorage类似.唯一的区别是,虽然存储在localStorage中的数据没有到期时间,但是浏览会话时存储在sessionStorage中的数据将被清除.结束-也就是说,当浏览器关闭时. n  n请注意,存储在localStorage或sessionStorage中的数据特定于页面协议.",
     },
-    "console": {
-      "assert": {
+    console: {
+      assert: {
         "!type": "fn(assertion: bool, text: string)",
         "!doc": "如果断言为false,则将错误消息写入控制台.",
       },
-      "error": {
+      error: {
         "!type": "fn(...msg: ?)",
         "!doc": "将错误消息输出到Web控制台.",
       },
-      "info": {
+      info: {
         "!type": "fn(...msg: ?)",
         "!doc": "将参考消息输出到Web控制台.",
       },
-      "log": {
+      log: {
         "!type": "fn(...msg: ?)",
         "!doc": "将消息输出到Web控制台.",
       },
-      "time": {
+      time: {
         "!type": "fn(label: string)",
         "!doc": "启动计时器,您可以使用该计时器来跟踪操作需要多长时间.",
       },
-      "timeEnd": {
+      timeEnd: {
         "!type": "fn(label: string)",
         "!doc": "停止以前通过调用console.time()启动的计时器.",
       },
-      "trace": {
+      trace: {
         "!type": "fn()",
         "!doc": "将堆栈跟踪输出到Web控制台.",
       },
-      "warn": {
+      warn: {
         "!type": "fn(...msg: ?)",
         "!doc": "将警告消息输出到Web控制台.",
       },
-      "!doc": "控制台对象提供对浏览器调试控制台的访问.其工作方式的细节因浏览器而异,但实际上提供了一组事实上的功能.",
+      "!doc":
+        "控制台对象提供对浏览器调试控制台的访问.其工作方式的细节因浏览器而异,但实际上提供了一组事实上的功能.",
     },
-    "window": {
+    window: {
       "!type": "<top>",
       "!doc": "窗口对象代表一个包含DOM文档的窗口.",
     },
-    "self": {
+    self: {
       "!type": "<top>",
       "!doc": "将对象引用返回到窗口对象.",
     },
-    "devicePixelRatio": "number",
-    "requestAnimationFrame": {
+    devicePixelRatio: "number",
+    requestAnimationFrame: {
       "!type": "fn(callback: fn(timestamp: number)) -> number",
-      "!doc": " Window.requestAnimationFrame()方法告诉浏览器您希望执行动画,并请求浏览器在下一次重绘之前调用指定的函数来更新动画.该方法将回调作为参数在重新粉刷之前被调用."
+      "!doc":
+        " Window.requestAnimationFrame()方法告诉浏览器您希望执行动画,并请求浏览器在下一次重绘之前调用指定的函数来更新动画.该方法将回调作为参数在重新粉刷之前被调用.",
     },
-    "cancelAnimationFrame": {
+    cancelAnimationFrame: {
       "!type": "fn(number)n",
       "!doc": "取消先前安排的动画帧请求.",
     },
-    "alert": {
+    alert: {
       "!type": "fn(message: string)",
-      "!doc": "显示具有指定内容和确定按钮的警报对话框."
+      "!doc": "显示具有指定内容和确定按钮的警报对话框.",
     },
-    "confirm": {
+    confirm: {
       "!type": "fn(message: string) -> bool",
       "!doc": "显示带有消息和两个按钮(确定和取消)的模式对话框.",
     },
-    "prompt": {
+    prompt: {
       "!type": "fn(message: string, value: string) -> string",
       "!doc": "显示一个对话框,提示用户输入一些文本.",
     },
-    "setTimeout": {
+    setTimeout: {
       "!type": "fn(f: fn(), ms: number) -> number",
-      "!doc": "在指定的延迟后调用函数或执行代码段."
+      "!doc": "在指定的延迟后调用函数或执行代码段.",
     },
-    "clearTimeout": {
+    clearTimeout: {
       "!type": "fn(timeout: number)",
       "!doc": "清除window.setTimeout()设置的延迟.",
     },
-    "setInterval": {
+    setInterval: {
       "!type": "fn(f: fn(), ms: number) -> number",
-      "!doc": "反复调用一个函数或执行代码段,每次调用该函数之间有固定的时间延迟.",
+      "!doc":
+        "反复调用一个函数或执行代码段,每次调用该函数之间有固定的时间延迟.",
     },
-    "clearInterval": {
+    clearInterval: {
       "!type": "fn(interval: number)",
       "!doc": "取消使用setInterval设置的重复操作.",
     },
-    "atob": {
+    atob: {
       "!type": "fn(encoded: string) -> string",
-      "!doc": "解码使用base-64编码编码的数据字符串."
+      "!doc": "解码使用base-64编码编码的数据字符串.",
     },
-    "btoa": {
+    btoa: {
       "!type": "fn(data: string) -> string",
       "!doc": "从一串二进制数据创建一个base-64编码的ASCII字符串.",
     },
-    "getComputedStyle": {
+    getComputedStyle: {
       "!type": "fn(node: +Element, pseudo?: string) -> Element.prototype.style",
       "!doc": "给出元素的所有CSS属性的最终使用值.",
     },
-    "CanvasRenderingContext2D": {
-      "canvas": "+Element",
-      "width": "number",
-      "height": "number",
-      "commit": "fn()",
-      "save": "fn()",
-      "restore": "fn()",
-      "currentTransform": "?",
-      "scale": "fn(x: number, y: number)",
-      "rotate": "fn(angle: number)",
-      "translate": "fn(x: number, y: number)",
-      "transform": "fn(a: number, b: number, c: number, d: number, e: number, f: number)",
-      "setTransform": "fn(a: number, b: number, c: number, d: number, e: number, f: number)",
-      "resetTransform": "fn()",
-      "globalAlpha": "number",
-      "globalCompositeOperation": "string",
-      "imageSmoothingEnabled": "bool",
-      "strokeStyle": "string",
-      "fillStyle": "string",
-      "createLinearGradient": "fn(x0: number, y0: number, x1: number, y1: number) -> ?",
-      "createPattern": "fn(image: ?, repetition: string) -> ?",
-      "shadowOffsetX": "number",
-      "shadowOffsetY": "number",
-      "shadowBlur": "number",
-      "shadowColor": "string",
-      "clearRect": "fn(x: number, y: number, w: number, h: number)",
-      "fillRect": "fn(x: number, y: number, w: number, h: number)",
-      "strokeRect": "fn(x: number, y: number, w: number, h: number)",
-      "fillRule": "string",
-      "fill": "fn()",
-      "beginPath": "fn()",
-      "stroke": "fn()",
-      "clip": "fn()",
-      "resetClip": "fn()",
-      "fillText": "fn(text: string, x: number, y: number, maxWidth: number)",
-      "strokeText": "fn(text: string, x: number, y: number, maxWidth: number)",
-      "measureText": "fn(text: string) -> ?",
-      "drawImage": "fn(image: ?, dx: number, dy: number)",
-      "createImageData": "fn(sw: number, sh: number) -> ?",
-      "getImageData": "fn(sx: number, sy: number, sw: number, sh: number) -> ?",
-      "putImageData": "fn(imagedata: ?, dx: number, dy: number)",
-      "lineWidth": "number",
-      "lineCap": "string",
-      "lineJoin": "string",
-      "miterLimit": "number",
-      "setLineDash": "fn(segments: [number])",
-      "getLineDash": "fn() -> [number]",
-      "lineDashOffset": "number",
-      "font": "string",
-      "textAlign": "string",
-      "textBaseline": "string",
-      "direction": "string",
-      "closePath": "fn()",
-      "moveTo": "fn(x: number, y: number)",
-      "lineTo": "fn(x: number, y: number)",
-      "quadraticCurveTo": "fn(cpx: number, cpy: number, x: number, y: number)",
-      "bezierCurveTo": "fn(cp1x: number, cp1y: number, cp2x: number, cp2y: number, x: number, y: number)",
-      "arcTo": "fn(x1: number, y1: number, x2: number, y2: number, radius: number)",
-      "rect": "fn(x: number, y: number, w: number, h: number)",
-      "arc": "fn(x: number, y: number, radius: number, startAngle: number, endAngle: number, anticlockwise?: bool)",
-      "ellipse": "fn(x: number, y: number, radiusX: number, radiusY: number, rotation: number, startAngle: number, endAngle: number, anticlockwise: bool)"
-    }
+    CanvasRenderingContext2D: {
+      canvas: "+Element",
+      width: "number",
+      height: "number",
+      commit: "fn()",
+      save: "fn()",
+      restore: "fn()",
+      currentTransform: "?",
+      scale: "fn(x: number, y: number)",
+      rotate: "fn(angle: number)",
+      translate: "fn(x: number, y: number)",
+      transform:
+        "fn(a: number, b: number, c: number, d: number, e: number, f: number)",
+      setTransform:
+        "fn(a: number, b: number, c: number, d: number, e: number, f: number)",
+      resetTransform: "fn()",
+      globalAlpha: "number",
+      globalCompositeOperation: "string",
+      imageSmoothingEnabled: "bool",
+      strokeStyle: "string",
+      fillStyle: "string",
+      createLinearGradient:
+        "fn(x0: number, y0: number, x1: number, y1: number) -> ?",
+      createPattern: "fn(image: ?, repetition: string) -> ?",
+      shadowOffsetX: "number",
+      shadowOffsetY: "number",
+      shadowBlur: "number",
+      shadowColor: "string",
+      clearRect: "fn(x: number, y: number, w: number, h: number)",
+      fillRect: "fn(x: number, y: number, w: number, h: number)",
+      strokeRect: "fn(x: number, y: number, w: number, h: number)",
+      fillRule: "string",
+      fill: "fn()",
+      beginPath: "fn()",
+      stroke: "fn()",
+      clip: "fn()",
+      resetClip: "fn()",
+      fillText: "fn(text: string, x: number, y: number, maxWidth: number)",
+      strokeText: "fn(text: string, x: number, y: number, maxWidth: number)",
+      measureText: "fn(text: string) -> ?",
+      drawImage: "fn(image: ?, dx: number, dy: number)",
+      createImageData: "fn(sw: number, sh: number) -> ?",
+      getImageData: "fn(sx: number, sy: number, sw: number, sh: number) -> ?",
+      putImageData: "fn(imagedata: ?, dx: number, dy: number)",
+      lineWidth: "number",
+      lineCap: "string",
+      lineJoin: "string",
+      miterLimit: "number",
+      setLineDash: "fn(segments: [number])",
+      getLineDash: "fn() -> [number]",
+      lineDashOffset: "number",
+      font: "string",
+      textAlign: "string",
+      textBaseline: "string",
+      direction: "string",
+      closePath: "fn()",
+      moveTo: "fn(x: number, y: number)",
+      lineTo: "fn(x: number, y: number)",
+      quadraticCurveTo: "fn(cpx: number, cpy: number, x: number, y: number)",
+      bezierCurveTo:
+        "fn(cp1x: number, cp1y: number, cp2x: number, cp2y: number, x: number, y: number)",
+      arcTo:
+        "fn(x1: number, y1: number, x2: number, y2: number, radius: number)",
+      rect: "fn(x: number, y: number, w: number, h: number)",
+      arc: "fn(x: number, y: number, radius: number, startAngle: number, endAngle: number, anticlockwise?: bool)",
+      ellipse:
+        "fn(x: number, y: number, radiusX: number, radiusY: number, rotation: number, startAngle: number, endAngle: number, anticlockwise: bool)",
+    },
   },
   {
     "!name": "ecmascript",
-    "Infinity": {
+    Infinity: {
       "!type": "number",
-      "!doc": "代表无穷大的数值."
+      "!doc": "代表无穷大的数值.",
     },
-    "undefined": {
+    undefined: {
       "!type": "?",
       "!doc": "该值未定义.",
     },
-    "NaN": {
+    NaN: {
       "!type": "number",
-      "!doc": "代表非数字的值."
+      "!doc": "代表非数字的值.",
     },
-    "Object": {
+    Object: {
       "!type": "fn()",
-      "create": {
+      create: {
         "!type": "fn(proto: ?) -> !custom:Object_create",
         "!doc": "使用指定的原型对象和属性创建一个新对象.",
       },
-      "defineProperty": {
-        "!type": "fn(obj: ?, prop: string, desc: propertyDescriptor) -> !custom:Object_defineProperty",
-        "!doc": "直接在对象上定义新属性,或修改对象上的现有属性,然后返回对象.如果想了解如何将Object.defineProperty方法与类似二进制标志的语法一起使用,请参阅本文."
+      defineProperty: {
+        "!type":
+          "fn(obj: ?, prop: string, desc: propertyDescriptor) -> !custom:Object_defineProperty",
+        "!doc":
+          "直接在对象上定义新属性,或修改对象上的现有属性,然后返回对象.如果想了解如何将Object.defineProperty方法与类似二进制标志的语法一起使用,请参阅本文.",
       },
-      "keys": {
+      keys: {
         "!type": "fn(obj: ?) -> [string]",
-        "!doc": "返回一个给定对象自己的可枚举属性的数组,其顺序与for-in循环所提供的顺序相同(不同之处在于for-in循环也枚举了原型链中的属性). "
+        "!doc":
+          "返回一个给定对象自己的可枚举属性的数组,其顺序与for-in循环所提供的顺序相同(不同之处在于for-in循环也枚举了原型链中的属性). ",
       },
-      "assign": {
+      assign: {
         "!type": "fn(target: ?, source: ?, source?: ?) -> !0",
         "!effects": ["copy !1 !0", "copy !2 !0", "copy !3 !0"],
-        "!doc": " Object.assign()方法用于将所有可枚举的自身属性的值从一个或多个源对象复制到目标对象.它将返回目标对象.,",
+        "!doc":
+          " Object.assign()方法用于将所有可枚举的自身属性的值从一个或多个源对象复制到目标对象.它将返回目标对象.,",
       },
-      "prototype": {
+      prototype: {
         "!stdProto": "Object",
-        "toString": {
+        toString: {
           "!type": "fn() -> string",
-          "!doc": "返回表示对象的字符串."
+          "!doc": "返回表示对象的字符串.",
         },
-        "hasOwnProperty": {
+        hasOwnProperty: {
           "!type": "fn(prop: string) -> bool",
           "!doc": "返回一个布尔值,指示对象是否具有指定的属性.",
-        }
+        },
       },
     },
-    "Function": {
+    Function: {
       "!type": "fn(body: string) -> fn()",
-      "prototype": {
+      prototype: {
         "!stdProto": "Function",
-        "apply": {
+        apply: {
           "!type": "fn(this: ?, args: [?])",
-          "!effects": [
-            "call and return !this this=!0 !1.<i> !1.<i> !1.<i>"
-          ],
-          "!doc": "调用具有给定值的函数,并以数组(或类似对象的数组)形式提供参数.",
+          "!effects": ["call and return !this this=!0 !1.<i> !1.<i> !1.<i>"],
+          "!doc":
+            "调用具有给定值的函数,并以数组(或类似对象的数组)形式提供参数.",
         },
-        "call": {
+        call: {
           "!type": "fn(this: ?, args?: ?) -> !this.!ret",
-          "!effects": [
-            "call and return !this this=!0 !1 !2 !3 !4"
-          ],
+          "!effects": ["call and return !this this=!0 !1 !2 !3 !4"],
           "!doc": "调用具有给定值和单独提供的参数的函数.",
         },
-        "bind": {
+        bind: {
           "!type": "fn(this: ?, args?: ?) -> !custom:Function_bind",
-          "!doc": "创建一个新函数,该函数在被调用时将其this关键字设置为提供的值,并在调用新函数时提供给定的参数序列.",
+          "!doc":
+            "创建一个新函数,该函数在被调用时将其this关键字设置为提供的值,并在调用新函数时提供给定的参数序列.",
         },
-        "prototype": "?"
+        prototype: "?",
       },
     },
-    "Array": {
+    Array: {
       "!type": "fn(size: number) -> !custom:Array_ctor",
-      "isArray": {
+      isArray: {
         "!type": "fn(value: ?) -> bool",
         "!doc": "如果对象是数组,则返回true,否则返回false.",
       },
-      "from": {
-        "!type": "fn(arrayLike: ?, mapFn?: fn(elt: ?, i: number) -> ?, thisArg?: ?) -> [!0.<i>]",
-        "!effects": [
-          "call !1 this=!2 !0.<i> number"
-        ],
-        "!doc": " Array.from()方法从类似数组或可迭代的对象创建一个新的Array实例.,",
+      from: {
+        "!type":
+          "fn(arrayLike: ?, mapFn?: fn(elt: ?, i: number) -> ?, thisArg?: ?) -> [!0.<i>]",
+        "!effects": ["call !1 this=!2 !0.<i> number"],
+        "!doc":
+          " Array.from()方法从类似数组或可迭代的对象创建一个新的Array实例.,",
       },
-      "of": {
+      of: {
         "!type": "fn(elementN: ?) -> [!0]",
-        "!doc": " Array.of()方法创建一个新的Array实例,该实例具有可变数量的参数,而不考虑参数的数量或类型.,",
+        "!doc":
+          " Array.of()方法创建一个新的Array实例,该实例具有可变数量的参数,而不考虑参数的数量或类型.,",
       },
-      "prototype": {
+      prototype: {
         "!stdProto": "Array",
-        "length": {
+        length: {
           "!type": "number",
           "!doc": "一个无符号的32位整数,指定数组中的元素数.",
         },
-        "concat": {
+        concat: {
           "!type": "fn(other: [?]) -> !this",
           "!doc": "返回一个新数组,该数组由该数组与其他数组和/或值组成.",
         },
-        "join": {
+        join: {
           "!type": "fn(separator?: string) -> string",
-          "!doc": "将数组的所有元素连接到字符串中."
+          "!doc": "将数组的所有元素连接到字符串中.",
         },
-        "splice": {
+        splice: {
           "!type": "fn(pos: number, amount: number, newelt?: ?) -> [?]",
           "!doc": "更改数组的内容,在删除旧元素的同时添加新元素.",
         },
-        "pop": {
+        pop: {
           "!type": "fn() -> !this.<i>",
           "!doc": "从数组中删除最后一个元素并返回该元素.",
         },
-        "push": {
+        push: {
           "!type": "fn(newelt: ?) -> number",
-          "!effects": [
-            "propagate !0 !this.<i>"
-          ],
+          "!effects": ["propagate !0 !this.<i>"],
           "!doc": "通过添加给定元素并返回数组的新长度来更改数组.",
         },
-        "shift": {
+        shift: {
           "!type": "fn() -> !this.<i>",
           "!doc": "从数组中删除第一个元素并返回该元素.此方法更改数组的长度.",
         },
-        "unshift": {
+        unshift: {
           "!type": "fn(newelt: ?) -> number",
-          "!effects": [
-            "propagate !0 !this.<i>"
-          ],
+          "!effects": ["propagate !0 !this.<i>"],
           "!doc": "将一个或多个元素添加到数组的开头,并返回数组的新长度.",
         },
-        "slice": {
+        slice: {
           "!type": "fn(from?: number, to?: number) -> !this",
-          "!doc": "返回数组一部分的浅表副本."
+          "!doc": "返回数组一部分的浅表副本.",
         },
-        "reverse": {
+        reverse: {
           "!type": "fn()",
-          "!doc": "就地反转数组.第一个数组元素变为最后一个,而最后一个数组变为第一个.",
+          "!doc":
+            "就地反转数组.第一个数组元素变为最后一个,而最后一个数组变为第一个.",
         },
-        "sort": {
+        sort: {
           "!type": "fn(compare?: fn(a: ?, b: ?) -> number)",
-          "!effects": [
-            "call !0 !this.<i> !this.<i>"
-          ],
-          "!doc": "将数组中的元素排序并返回数组."
+          "!effects": ["call !0 !this.<i> !this.<i>"],
+          "!doc": "将数组中的元素排序并返回数组.",
         },
-        "indexOf": {
+        indexOf: {
           "!type": "fn(elt: ?, from?: number) -> number",
-          "!doc": "返回在数组中可以找到给定元素的第一个索引;如果不存在,则返回-1.",
+          "!doc":
+            "返回在数组中可以找到给定元素的第一个索引;如果不存在,则返回-1.",
         },
-        "lastIndexOf": {
+        lastIndexOf: {
           "!type": "fn(elt: ?, from?: number) -> number",
-          "!doc": "返回在数组中找到给定元素的最后一个索引,如果不存在则返回-1.从fromIndex开始向后搜索数组.",
+          "!doc":
+            "返回在数组中找到给定元素的最后一个索引,如果不存在则返回-1.从fromIndex开始向后搜索数组.",
         },
-        "filter": {
-          "!type": "fn(test: fn(elt: ?, i: number, array: +Array) -> bool, context?: ?) -> !this",
-          "!effects": [
-            "call !0 this=!1 !this.<i> number !this"
-          ],
+        filter: {
+          "!type":
+            "fn(test: fn(elt: ?, i: number, array: +Array) -> bool, context?: ?) -> !this",
+          "!effects": ["call !0 this=!1 !this.<i> number !this"],
           "!doc": "创建一个新数组,其中包含所有通过提供的功能实现的测试的元素.",
         },
-        "forEach": {
+        forEach: {
           "!type": "fn(f: fn(elt: ?, i: number, array: +Array), context?: ?)",
-          "!effects": [
-            "call !0 this=!1 !this.<i> number !this"
-          ],
-          "!doc": "每个数组元素执行一次提供的功能."
+          "!effects": ["call !0 this=!1 !this.<i> number !this"],
+          "!doc": "每个数组元素执行一次提供的功能.",
         },
-        "map": {
-          "!type": "fn(f: fn(elt: ?, i: number, array: +Array) -> ?, context?: ?) -> [!0.!ret]",
-          "!effects": [
-            "call !0 this=!1 !this.<i> number !this"
-          ],
+        map: {
+          "!type":
+            "fn(f: fn(elt: ?, i: number, array: +Array) -> ?, context?: ?) -> [!0.!ret]",
+          "!effects": ["call !0 this=!1 !this.<i> number !this"],
           "!doc": "创建一个新数组,其结果是对该数组中的每个元素调用提供的函数.",
         },
-        "reduce": {
-          "!type": "fn(combine: fn(sum: ?, elt: ?, i: number, array: +Array) -> ?, init?: ?) -> !0.!ret",
-          "!effects": [
-            "call !0 !1 !this.<i> number !this"
-          ],
-          "!doc": "对一个累加器和数组的每个值(从左到右)应用一个函数,以将其减小为单个值.",
+        reduce: {
+          "!type":
+            "fn(combine: fn(sum: ?, elt: ?, i: number, array: +Array) -> ?, init?: ?) -> !0.!ret",
+          "!effects": ["call !0 !1 !this.<i> number !this"],
+          "!doc":
+            "对一个累加器和数组的每个值(从左到右)应用一个函数,以将其减小为单个值.",
         },
-        "fill": {
+        fill: {
           "!type": "fn(value: ?, start?: number, end?: number) -> !this",
-          "!doc": " fill()方法使用静态值填充数组的所有元素,从开始索引到结束索引.,",
+          "!doc":
+            " fill()方法使用静态值填充数组的所有元素,从开始索引到结束索引.,",
         },
-        "find": {
-          "!type": "fn(callback: fn(element: ?, index: number, array: [?]) -> bool, thisArg?: ?) -> !this.<i>",
+        find: {
+          "!type":
+            "fn(callback: fn(element: ?, index: number, array: [?]) -> bool, thisArg?: ?) -> !this.<i>",
           "!effects": ["call !0 this=!2 !this.<i> number"],
-          "!doc": "如果数组中的元素满足提供的测试功能,则find()方法将在数组中返回一个值.否则,返回undefined.,",
+          "!doc":
+            "如果数组中的元素满足提供的测试功能,则find()方法将在数组中返回一个值.否则,返回undefined.,",
         },
-        "findIndex": {
-          "!type": "fn(callback: fn(element: ?, index: number, array: [?]), thisArg?: ?) -> number",
+        findIndex: {
+          "!type":
+            "fn(callback: fn(element: ?, index: number, array: [?]), thisArg?: ?) -> number",
           "!effects": ["call !0 this=!2 !this.<i> number"],
-          "!doc": "如果数组中的元素满足提供的测试功能,则findIndex()方法将返回数组中的索引.否则返回-1.,",
+          "!doc":
+            "如果数组中的元素满足提供的测试功能,则findIndex()方法将返回数组中的索引.否则返回-1.,",
         },
-        "keys": {
+        keys: {
           "!type": "fn() -> +iter[:t=number]",
-          "!doc": " keys()方法返回一个新的数组迭代器,其中包含数组中每个索引的键.,",
+          "!doc":
+            " keys()方法返回一个新的数组迭代器,其中包含数组中每个索引的键.,",
         },
-        "values": {
+        values: {
           "!type": "fn() -> +iter[:t=!this.<i>]",
-          "!doc": " values()方法返回一个新的Array Iterator对象,该对象包含数组中每个索引的值.,",
+          "!doc":
+            " values()方法返回一个新的Array Iterator对象,该对象包含数组中每个索引的值.,",
         },
-        "includes": {
+        includes: {
           "!type": "fn(value: ?, fromIndex?: number) -> bool",
           "!doc": "确定数组是否包含某个元素,并根据需要返回true或false.,",
-        }
+        },
       },
     },
-    "String": {
+    String: {
       "!type": "fn(value: ?) -> string",
-      "prototype": {
+      prototype: {
         "!stdProto": "String",
-        "length": {
+        length: {
           "!type": "number",
-          "!doc": "表示字符串的长度."
+          "!doc": "表示字符串的长度.",
         },
         "<i>": "string",
-        "charAt": {
+        charAt: {
           "!type": "fn(i: number) -> string",
           "!doc": "从字符串中返回指定的字符.",
         },
-        "charCodeAt": {
+        charCodeAt: {
           "!type": "fn(i: number) -> number",
-          "!doc": "返回给定索引处字符的数字Unicode值(Unicode代码点> 0x10000除外).",
+          "!doc":
+            "返回给定索引处字符的数字Unicode值(Unicode代码点> 0x10000除外).",
         },
-        "indexOf": {
+        indexOf: {
           "!type": "fn(char: string, from?: number) -> number",
-          "!doc": "返回指定值首次出现的调用String对象中的索引,从fromIndex开始搜索,\ n如果未找到该值,则返回-1.",
+          "!doc":
+            "返回指定值首次出现的调用String对象中的索引,从fromIndex开始搜索, n如果未找到该值,则返回-1.",
         },
-        "lastIndexOf": {
+        lastIndexOf: {
           "!type": "fn(char: string, from?: number) -> number",
-          "!doc": "返回指定值最后一次出现的调用String对象内的索引,如果未找到则返回-1.从fromIndex开始向后搜索调用字符串.",
+          "!doc":
+            "返回指定值最后一次出现的调用String对象内的索引,如果未找到则返回-1.从fromIndex开始向后搜索调用字符串.",
         },
-        "substring": {
+        substring: {
           "!type": "fn(from: number, to?: number) -> string",
-          "!doc": "返回一个索引与另一个索引之间或字符串末尾的字符串子集.<br/>from为起始位置,to为终止位置.",
+          "!doc":
+            "返回一个索引与另一个索引之间或字符串末尾的字符串子集.<br/>from为起始位置,to为终止位置.",
         },
-        "substr": {
+        substr: {
           "!type": "fn(from: number, length?: number) -> string",
-          "!doc": "以指定的字符数返回从指定位置开始的字符串中的字符.<br/>from为起始位置,length为长度",
+          "!doc":
+            "以指定的字符数返回从指定位置开始的字符串中的字符.<br/>from为起始位置,length为长度",
         },
-        "slice": {
+        slice: {
           "!type": "fn(from: number, to?: number) -> string",
           "!doc": "提取字符串的一部分并返回新的字符串.",
         },
-        "padStart": {
+        padStart: {
           "!type": "fn(targetLength: number, padString?: string) -> string",
-          "!doc": "用另一个字符串(如果需要,重复)填充当前字符串,以使结果字符串达到给定的长度.",
+          "!doc":
+            "用另一个字符串(如果需要,重复)填充当前字符串,以使结果字符串达到给定的长度.",
         },
-        "padEnd": {
+        padEnd: {
           "!type": "fn(targetLength: number, padString?: string) -> string",
-          "!doc": "用给定的字符串(如果需要,重复)填充当前字符串,以使结果字符串达到给定的长度.",
+          "!doc":
+            "用给定的字符串(如果需要,重复)填充当前字符串,以使结果字符串达到给定的长度.",
         },
-        "trim": {
+        trim: {
           "!type": "fn() -> string",
           "!doc": "从字符串的两端删除空格.",
         },
-        "trimStart": {
+        trimStart: {
           "!type": "fn() -> string",
           "!doc": "从字符串的开头删除空格.",
         },
-        "trimEnd": {
+        trimEnd: {
           "!type": "fn() -> string",
           "!doc": "从字符串末尾删除空格.",
         },
-        "toUpperCase": {
+        toUpperCase: {
           "!type": "fn() -> string",
-          "!doc": "返回转换为大写的调用字符串值."
+          "!doc": "返回转换为大写的调用字符串值.",
         },
-        "toLowerCase": {
+        toLowerCase: {
           "!type": "fn() -> string",
-          "!doc": "返回转换为小写的调用字符串值."
+          "!doc": "返回转换为小写的调用字符串值.",
         },
-        "split": {
+        split: {
           "!type": "fn(pattern?: string|+RegExp, limit?: number) -> [string]",
           "!doc": "通过将字符串分成子字符串,将String对象拆分为字符串数组.",
         },
-        "concat": {
+        concat: {
           "!type": "fn(other: string) -> string",
-          "!doc": "将两个或多个字符串的文本合并,并返回一个新字符串."
+          "!doc": "将两个或多个字符串的文本合并,并返回一个新字符串.",
         },
-        "match": {
+        match: {
           "!type": "fn(pattern: +RegExp) -> [string]",
           "!doc": "用于将字符串与正则表达式匹配时用于检索匹配.",
         },
-        "replace": {
+        replace: {
           "!type": "fn(pattern: string|+RegExp, replacement: string) -> string",
-          "!doc": "返回一个新字符串,该字符串的某个或所有匹配项都由替换项替换.该模式可以是字符串或RegExp,并且替换项可以是字符串或每个匹配项将调用的函数. "
+          "!doc":
+            "返回一个新字符串,该字符串的某个或所有匹配项都由替换项替换.该模式可以是字符串或RegExp,并且替换项可以是字符串或每个匹配项将调用的函数. ",
         },
-        "endsWith": {
+        endsWith: {
           "!type": "fn(searchString: string, position?: number) -> bool",
-          "!doc": " endsWith()方法确定一个字符串是否以另一个字符串的字符结尾,并根据需要返回true或false.,",
+          "!doc":
+            " endsWith()方法确定一个字符串是否以另一个字符串的字符结尾,并根据需要返回true或false.,",
         },
-        "startsWith": {
+        startsWith: {
           "!type": "fn(searchString: string, position?: number) -> bool",
-          "!doc": " startsWith()方法确定一个字符串是否以另一个字符串的字符开头,并根据需要返回true或false.,",
-        }
+          "!doc":
+            " startsWith()方法确定一个字符串是否以另一个字符串的字符开头,并根据需要返回true或false.,",
+        },
       },
     },
-    "Number": {
+    Number: {
       "!type": "fn(value: ?) -> number",
-      "MAX_VALUE": {
+      MAX_VALUE: {
         "!type": "number",
-        "!doc": " JavaScript中可表示的最大数值."
+        "!doc": " JavaScript中可表示的最大数值.",
       },
-      "MIN_VALUE": {
+      MIN_VALUE: {
         "!type": "number",
-        "!doc": " JavaScript中可表示的最小正数值."
+        "!doc": " JavaScript中可表示的最小正数值.",
       },
-      "POSITIVE_INFINITY": {
+      POSITIVE_INFINITY: {
         "!type": "number",
-        "!doc": "代表正无穷大值的值."
+        "!doc": "代表正无穷大值的值.",
       },
-      "NEGATIVE_INFINITY": {
+      NEGATIVE_INFINITY: {
         "!type": "number",
-        "!doc": "代表负无穷大值的值."
+        "!doc": "代表负无穷大值的值.",
       },
-      "prototype": {
+      prototype: {
         "!stdProto": "Number",
-        "toString": {
+        toString: {
           "!type": "fn(radix?: number) -> string",
-          "!doc": "返回代表指定Number对象的字符串"
+          "!doc": "返回代表指定Number对象的字符串",
         },
-        "toFixed": {
+        toFixed: {
           "!type": "fn(digits: number) -> string",
-          "!doc": "使用定点符号格式化数字"
+          "!doc": "使用定点符号格式化数字",
         },
-        "toExponential": {
+        toExponential: {
           "!type": "fn(digits: number) -> string",
-          "!doc": "返回以指数表示形式表示Number对象的字符串"
+          "!doc": "返回以指数表示形式表示Number对象的字符串",
         },
-        "toPrecision": {
+        toPrecision: {
           "!type": "fn(digits: number) -> string",
           "!doc": " toPrecision()方法返回一个字符串,该数字表示指定精度的数字.",
-        }
+        },
       },
-      "EPSILON": {
+      EPSILON: {
         "!type": "number",
-        "!doc": " Number.EPSILON属性表示一个数值与可以表示为Number的最小值之间的差异.,",
+        "!doc":
+          " Number.EPSILON属性表示一个数值与可以表示为Number的最小值之间的差异.,",
       },
-      "MAX_SAFE_INTEGER": {
+      MAX_SAFE_INTEGER: {
         "!type": "number",
-        "!doc": " Number.MAX_SAFE_INTEGER常量表示JavaScript中的最大安全整数(2 ^ 53-1).,",
+        "!doc":
+          " Number.MAX_SAFE_INTEGER常量表示JavaScript中的最大安全整数(2 ^ 53-1).,",
       },
-      "MIN_SAFE_INTEGER": {
+      MIN_SAFE_INTEGER: {
         "!type": "number",
-        "!doc": " Number.MIN_SAFE_INTEGER常量表示JavaScript(-(2 ^ 53-1))中的最小安全整数.,",
+        "!doc":
+          " Number.MIN_SAFE_INTEGER常量表示JavaScript(-(2 ^ 53-1))中的最小安全整数.,",
       },
-      "isFinite": {
+      isFinite: {
         "!type": "fn(testValue: ?) -> bool",
         "!doc": " Number.isFinite()方法确定传递的值是否为有限值.,",
       },
-      "isInteger": {
+      isInteger: {
         "!type": "fn(testValue: ?) -> bool",
         "!doc": " Number.isInteger()方法确定传递的值是否为整数.,",
       },
-      "isNaN": {
+      isNaN: {
         "!type": "fn(testValue: ?) -> bool",
-        "!doc": " Number.isNaN()方法确定传递的值是否为NaN.原始全局isNaN()的更可靠的版本.,",
+        "!doc":
+          " Number.isNaN()方法确定传递的值是否为NaN.原始全局isNaN()的更可靠的版本.,",
       },
-      "isSafeInteger": {
+      isSafeInteger: {
         "!type": "fn(testValue: ?) -> bool",
-        "!doc": " Number.isSafeInteger()方法确定所提供的值是否是一个安全整数的数字.安全整数是该数字的整数.",
+        "!doc":
+          " Number.isSafeInteger()方法确定所提供的值是否是一个安全整数的数字.安全整数是该数字的整数.",
       },
-      "parseFloat": {
+      parseFloat: {
         "!type": "fn(string: string) -> number",
         "!doc": " Number.parseFloat()方法解析字符串参数并返回浮点数.,",
       },
-      "parseInt": {
+      parseInt: {
         "!type": "fn(string: string, radix?: number) -> number",
-        "!doc": " Number.parseInt()方法解析字符串参数并返回指定基数或基数的整数.,",
+        "!doc":
+          " Number.parseInt()方法解析字符串参数并返回指定基数或基数的整数.,",
       },
     },
-    "Boolean": {
+    Boolean: {
       "!type": "fn(value: ?) -> bool",
-      "prototype": {
-        "!stdProto": "Boolean"
+      prototype: {
+        "!stdProto": "Boolean",
       },
     },
-    "abstract": "?",
-    "arguments": "?",
-    "boolean": "?",
-    "break": "?",
-    "byte": "?",
-    "case": "?",
-    "catch": "?",
-    "char": "?",
-    "const": "?",
-    "continue": "?",
-    "debugger": "?",
-    "default": "?",
-    "delete": "?",
-    "do": "?",
-    "double": "?",
-    "else": "?",
-    "eval": "?",
-    "false": "bool",
-    "final": "?",
-    "finally": "?",
-    "float": "?",
-    "for": "?",
-    "function": "?",
-    "goto": "?",
-    "if": "?",
-    "implements": "?",
-    "in": "?",
-    "instanceof": "?",
-    "int": "?",
-    "interface": "?",
-    "long": "?",
-    "native": "?",
-    "new": "?",
-    "null": "?",
-    "package": "?",
-    "private": "?",
-    "protected": "?",
-    "public": "?",
-    "return": "?",
-    "short": "?",
-    "static": "?",
-    "switch": "?",
-    "synchronized": "?",
-    "this": "?",
-    "throw": "?",
-    "throws": "?",
-    "transient": "?",
-    "true": "bool",
-    "try": "?",
-    "typeof": "?",
-    "var": "?",
-    "void": "?",
-    "volatile": "?",
-    "while": "?",
-    "with": "?",
-    "yield": "?",
-    "RegExp": {
+    abstract: "?",
+    arguments: "?",
+    boolean: "?",
+    break: "?",
+    byte: "?",
+    case: "?",
+    catch: "?",
+    char: "?",
+    const: "?",
+    continue: "?",
+    debugger: "?",
+    default: "?",
+    delete: "?",
+    do: "?",
+    double: "?",
+    else: "?",
+    eval: "?",
+    false: "bool",
+    final: "?",
+    finally: "?",
+    float: "?",
+    for: "?",
+    function: "?",
+    goto: "?",
+    if: "?",
+    implements: "?",
+    in: "?",
+    instanceof: "?",
+    int: "?",
+    interface: "?",
+    long: "?",
+    native: "?",
+    new: "?",
+    null: "?",
+    package: "?",
+    private: "?",
+    protected: "?",
+    public: "?",
+    return: "?",
+    short: "?",
+    static: "?",
+    switch: "?",
+    synchronized: "?",
+    this: "?",
+    throw: "?",
+    throws: "?",
+    transient: "?",
+    true: "bool",
+    try: "?",
+    typeof: "?",
+    var: "?",
+    void: "?",
+    volatile: "?",
+    while: "?",
+    with: "?",
+    yield: "?",
+    RegExp: {
       "!type": "fn(source: string, flags?: string)",
-      "prototype": {
+      prototype: {
         "!stdProto": "RegExp",
-        "exec": {
+        exec: {
           "!type": "fn(input: string) -> [string]",
           "!doc": "搜索指定字符串中的匹配项.返回结果数组,或者为null.",
         },
-        "test": {
+        test: {
           "!type": "fn(input: string) -> bool",
           "!doc": "执行正则表达式和指定字符串之间的匹配搜索.返回true或false.",
-        }
+        },
       },
       "!doc": "创建正则表达式对象以将文本与模式匹配.",
     },
-    "parseInt": {
+    parseInt: {
       "!type": "fn(string: string, radix?: number) -> number",
-      "!doc": "解析字符串参数并返回指定基数或基数的整数."
+      "!doc": "解析字符串参数并返回指定基数或基数的整数.",
     },
-    "parseFloat": {
+    parseFloat: {
       "!type": "fn(string: string) -> number",
-      "!doc": "解析字符串参数并返回浮点数."
+      "!doc": "解析字符串参数并返回浮点数.",
     },
-    "isNaN": {
+    isNaN: {
       "!type": "fn(value: number) -> bool",
-      "!doc": "确定值是否为NaN.请注意,此函数已损坏.您可能对ECMAScript 6 Number.isNaN感兴趣.",
+      "!doc":
+        "确定值是否为NaN.请注意,此函数已损坏.您可能对ECMAScript 6 Number.isNaN感兴趣.",
     },
-    "isFinite": {
+    isFinite: {
       "!type": "fn(value: number) -> bool",
-      "!doc": "确定传递的值是否为有限数字."
+      "!doc": "确定传递的值是否为有限数字.",
     },
-    "eval": {
+    eval: {
       "!type": "fn(code: string) -> ?",
-      "!doc": "评估以字符串形式表示的JavaScript代码."
+      "!doc": "评估以字符串形式表示的JavaScript代码.",
     },
-    "encodeURI": {
+    encodeURI: {
       "!type": "fn(uri: string) -> string",
-      "!doc": "通过用表示字符的UTF-8编码的一个,两个,三个或四个转义序列替换某些字符的每个实例来编码统一资源标识符(URI)(对于字符而言将仅是四个转义序列由两个\"代理\"字符组成).",
+      "!doc":
+        '通过用表示字符的UTF-8编码的一个,两个,三个或四个转义序列替换某些字符的每个实例来编码统一资源标识符(URI)(对于字符而言将仅是四个转义序列由两个"代理"字符组成).',
     },
-    "encodeURIComponent": {
+    encodeURIComponent: {
       "!type": "fn(uri: string) -> string",
-      "!doc": "通过用表示字符的UTF-8编码的一个,两个,三个或四个转义序列替换某些字符的每个实例来编码统一资源标识符(URI)组件(对于由两个\"代理\"字符组成的字符).",
+      "!doc":
+        '通过用表示字符的UTF-8编码的一个,两个,三个或四个转义序列替换某些字符的每个实例来编码统一资源标识符(URI)组件(对于由两个"代理"字符组成的字符).',
     },
-    "decodeURI": {
+    decodeURI: {
       "!type": "fn(uri: string) -> string",
       "!doc": "解码以前由encodeURI或类似例程创建的统一资源标识符(URI).",
     },
-    "decodeURIComponent": {
+    decodeURIComponent: {
       "!type": "fn(uri: string) -> string",
-      "!doc": "解码以前由encodeURIComponent或类似例程创建的统一资源标识符(URI)组件.",
+      "!doc":
+        "解码以前由encodeURIComponent或类似例程创建的统一资源标识符(URI)组件.",
     },
-    "Math": {
-      "E": {
+    Math: {
+      E: {
         "!type": "number",
-        "!doc": "自然对数的底数,e约为2.718."
+        "!doc": "自然对数的底数,e约为2.718.",
       },
-      "LN2": {
+      LN2: {
         "!type": "number",
-        "!doc": " 2的自然对数,大约为0.693."
+        "!doc": " 2的自然对数,大约为0.693.",
       },
-      "LN10": {
+      LN10: {
         "!type": "number",
-        "!doc": " 10的自然对数,大约为2.302."
+        "!doc": " 10的自然对数,大约为2.302.",
       },
-      "LOG2E": {
+      LOG2E: {
         "!type": "number",
         "!doc": " E的以2为底的对数(大约1.442).",
       },
-      "LOG10E": {
+      LOG10E: {
         "!type": "number",
-        "!doc": " E的以10为底的对数(约0.434)."
+        "!doc": " E的以10为底的对数(约0.434).",
       },
-      "SQRT1_2": {
+      SQRT1_2: {
         "!type": "number",
-        "!doc": " 1/2的平方根;等效于2的平方根上的1,大约为0.707."
+        "!doc": " 1/2的平方根;等效于2的平方根上的1,大约为0.707.",
       },
-      "SQRT2": {
+      SQRT2: {
         "!type": "number",
-        "!doc": " 2的平方根,大约为1.414."
+        "!doc": " 2的平方根,大约为1.414.",
       },
-      "PI": {
+      PI: {
         "!type": "number",
-        "!doc": "圆的周长与其直径之比,大约为3.14159."
+        "!doc": "圆的周长与其直径之比,大约为3.14159.",
       },
-      "abs": {
+      abs: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的绝对值."
+        "!doc": "返回数字的绝对值.",
       },
-      "cos": {
+      cos: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的余弦."
+        "!doc": "返回数字的余弦.",
       },
-      "sin": {
+      sin: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的正弦."
+        "!doc": "返回数字的正弦.",
       },
-      "tan": {
+      tan: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的正切值."
+        "!doc": "返回数字的正切值.",
       },
-      "acos": {
+      acos: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的反余弦(以弧度为单位)."
+        "!doc": "返回数字的反余弦(以弧度为单位).",
       },
-      "asin": {
+      asin: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的反正弦(以弧度为单位)."
+        "!doc": "返回数字的反正弦(以弧度为单位).",
       },
-      "atan": {
+      atan: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的反正切(以弧度为单位)."
+        "!doc": "返回数字的反正切(以弧度为单位).",
       },
-      "atan2": {
+      atan2: {
         "!type": "fn(y: number, x: number) -> number",
-        "!doc": "返回其参数商的反正切值."
+        "!doc": "返回其参数商的反正切值.",
       },
-      "ceil": {
+      ceil: {
         "!type": "fn(number) -> number",
-        "!doc": "返回大于或等于数字的最小整数."
+        "!doc": "返回大于或等于数字的最小整数.",
       },
-      "floor": {
+      floor: {
         "!type": "fn(number) -> number",
-        "!doc": "返回小于或等于数字的最大整数."
+        "!doc": "返回小于或等于数字的最大整数.",
       },
-      "round": {
+      round: {
         "!type": "fn(number) -> number",
-        "!doc": "返回四舍五入到最接近整数的数字的值."
+        "!doc": "返回四舍五入到最接近整数的数字的值.",
       },
-      "exp": {
+      exp: {
         "!type": "fn(number) -> number",
-        "!doc": "返回E ^ x,其中x是自变量,E是欧拉常数,自然对数的底."
+        "!doc": "返回E ^ x,其中x是自变量,E是欧拉常数,自然对数的底.",
       },
-      "log": {
+      log: {
         "!type": "fn(number) -> number",
         "!doc": "返回数字的自然对数(以E为底).",
       },
-      "sqrt": {
+      sqrt: {
         "!type": "fn(number) -> number",
-        "!doc": "返回数字的平方根."
+        "!doc": "返回数字的平方根.",
       },
-      "pow": {
+      pow: {
         "!type": "fn(number, number) -> number",
-        "!doc": "将基数返回指数幂,即baseexponent."
+        "!doc": "将基数返回指数幂,即baseexponent.",
       },
-      "max": {
+      max: {
         "!type": "fn(number, number) -> number",
-        "!doc": "返回零个或多个数字中的最大值."
+        "!doc": "返回零个或多个数字中的最大值.",
       },
-      "min": {
+      min: {
         "!type": "fn(number, number) -> number",
-        "!doc": "返回零个或多个数字中的最小值."
+        "!doc": "返回零个或多个数字中的最小值.",
       },
-      "random": {
+      random: {
         "!type": "fn() -> number",
-        "!doc": "返回一个浮点伪随机数,范围为[0,1),即从0(包括)到不包括1(排除),然后您可以缩放到所需的值范围."
+        "!doc":
+          "返回一个浮点伪随机数,范围为[0,1),即从0(包括)到不包括1(排除),然后您可以缩放到所需的值范围.",
       },
-      "log10": {
+      log10: {
         "!type": "fn(x: number) -> number",
         "!doc": " Math.log10()函数返回数字的以10为底的对数.",
       },
-      "log2": {
+      log2: {
         "!type": "fn(x: number) -> number",
         "!doc": " Math.log2()函数返回数字的以2为底的对数.",
       },
-      "sign": {
+      sign: {
         "!type": "fn(x: number) -> number",
         "!doc": " Math.sign()函数返回数字的符号,指示数字是正数,负数还是零.,",
       },
-      "trunc": {
+      trunc: {
         "!type": "fn(x: number) -> number",
-        "!doc": " Math.trunc()函数通过删除任何小数位来返回数字的整数部分.它不舍入任何数字.该函数可以用floor()和ceil()函数表示: ,",
+        "!doc":
+          " Math.trunc()函数通过删除任何小数位来返回数字的整数部分.它不舍入任何数字.该函数可以用floor()和ceil()函数表示: ,",
       },
       "!doc": "一个内置对象,具有用于数学常数和函数的属性和方法.",
     },
-    "JSON": {
-      "parse": {
-        "!type": "fn(json: string, reviver?: fn(key: string, value: ?) -> ?) -> ?",
+    JSON: {
+      parse: {
+        "!type":
+          "fn(json: string, reviver?: fn(key: string, value: ?) -> ?) -> ?",
         "!doc": "将字符串解析为JSON,可以选择转换解析产生的值.",
       },
-      "stringify": {
-        "!type": "fn(value: ?, replacer?: fn(key: string, value: ?) -> ?, space?: string|number) -> string",
-        "!doc": "将值转换为JSON,如果指定了replacer函数,则可以选择替换值,如果指定了replacer数组,则可以选择仅包括指定的属性.",
+      stringify: {
+        "!type":
+          "fn(value: ?, replacer?: fn(key: string, value: ?) -> ?, space?: string|number) -> string",
+        "!doc":
+          "将值转换为JSON,如果指定了replacer函数,则可以选择替换值,如果指定了replacer数组,则可以选择仅包括指定的属性.",
       },
-      "!doc": " JSON(JavaScript对象表示法)是一种数据交换格式.尽管它不是严格的子集,但它非常类似于JavaScript语法的子集.(有关详细信息,请参见JavaScript参考中的JSON.)在编写任何类型的基于JavaScript的应用程序(包括网站和浏览器扩展程序)时非常有用.例如,您可以将JSON格式的用户信息存储在cookie中,或者可以将扩展名首选项以JSON形式存储在字符串值的浏览器首选项中."
-    }
+      "!doc":
+        " JSON(JavaScript对象表示法)是一种数据交换格式.尽管它不是严格的子集,但它非常类似于JavaScript语法的子集.(有关详细信息,请参见JavaScript参考中的JSON.)在编写任何类型的基于JavaScript的应用程序(包括网站和浏览器扩展程序)时非常有用.例如,您可以将JSON格式的用户信息存储在cookie中,或者可以将扩展名首选项以JSON形式存储在字符串值的浏览器首选项中.",
+    },
   },
   {
     "!name": "core",
     "!define": {
-      "image": {
+      image: {
         "!doc": "图片信息",
-        "width": "number",
-        "height": "number",
-        "src": "string"
+        width: "number",
+        height: "number",
+        src: "string",
       },
-      "audio": {
+      audio: {
         "!doc": "音乐音效信息",
-        "currentTime": "number",
-        "play": "fn()",
-        "pause": "fn()",
-        "paused": "bool",
-        "duration": "number",
-        "volume": "number",
+        currentTime: "number",
+        play: "fn()",
+        pause: "fn()",
+        paused: "bool",
+        duration: "number",
+        volume: "number",
       },
-      "flag": {
+      flag: {
         "!doc": "当前变量",
-        "hard": {
+        hard: {
           "!type": "number",
-          "!doc": "当前难度编号"
+          "!doc": "当前难度编号",
         },
-        "hatred": {
+        hatred: {
           "!type": "number",
-          "!doc": "当前仇恨值"
+          "!doc": "当前仇恨值",
         },
-        "poison": {
+        poison: {
           "!type": "bool",
-          "!doc": "是否处于中毒状态"
+          "!doc": "是否处于中毒状态",
         },
-        "weak": {
+        weak: {
           "!type": "number",
-          "!doc": "是否处于衰弱状态"
+          "!doc": "是否处于衰弱状态",
         },
-        "curse": {
+        curse: {
           "!type": "number",
-          "!doc": "是否处于诅咒状态"
+          "!doc": "是否处于诅咒状态",
         },
-        "no_zone": {
+        no_zone: {
           "!type": "bool",
-          "!doc": "无视领域伤害"
+          "!doc": "无视领域伤害",
         },
-        "no_repulse": {
+        no_repulse: {
           "!type": "bool",
-          "!doc": "无视阻击伤害"
+          "!doc": "无视阻击伤害",
         },
-        "no_lasel": {
+        no_lasel: {
           "!type": "bool",
-          "!doc": "无视激光伤害"
+          "!doc": "无视激光伤害",
         },
-        "no_ambush": {
+        no_ambush: {
           "!type": "bool",
-          "!doc": "无视捕捉"
+          "!doc": "无视捕捉",
         },
-        "__bgm__": {
+        __bgm__: {
           "!type": "string",
-          "!doc": "背景音乐"
+          "!doc": "背景音乐",
         },
-        "__weather__": {
-          "!doc": "天气"
+        __weather__: {
+          "!doc": "天气",
         },
-        "__color__": {
-          "!doc": "色调"
+        __color__: {
+          "!doc": "色调",
         },
-        "__volume__": {
+        __volume__: {
           "!type": "number",
-          "!doc": "音量"
+          "!doc": "音量",
         },
-        "skill": {
+        skill: {
           "!type": "number",
-          "!doc": "当前开启的技能编号"
+          "!doc": "当前开启的技能编号",
         },
-        "skillName": {
+        skillName: {
           "!type": "string",
-          "!doc": "当前开启的技能名"
+          "!doc": "当前开启的技能名",
         },
-        "input": {
+        input: {
           "!type": "string|number",
-          "!doc": "等待用户输入后的存放值"
+          "!doc": "等待用户输入后的存放值",
         },
-        "type": {
+        type: {
           "!type": "number",
-          "!doc": "等待用户操作后获得的操作类型"
+          "!doc": "等待用户操作后获得的操作类型",
         },
-        "keycode": {
+        keycode: {
           "!type": "number",
-          "!doc": "等待用户操作后用户按键的键值"
+          "!doc": "等待用户操作后用户按键的键值",
         },
-        "x": {
+        x: {
           "!type": "number",
-          "!doc": "等待用户操作后用户点击的网格横坐标"
+          "!doc": "等待用户操作后用户点击的网格横坐标",
         },
-        "y": {
+        y: {
           "!type": "number",
-          "!doc": "等待用户操作后用户点击的网格纵坐标"
+          "!doc": "等待用户操作后用户点击的网格纵坐标",
         },
-        "px": {
+        px: {
           "!type": "number",
-          "!doc": "等待用户操作后用户点击的像素横坐标"
+          "!doc": "等待用户操作后用户点击的像素横坐标",
         },
-        "py": {
+        py: {
           "!type": "number",
-          "!doc": "等待用户操作后用户点击的像素纵坐标"
+          "!doc": "等待用户操作后用户点击的像素纵坐标",
         },
-        "__visited__": {
-          "!doc": "当前访问过的楼层"
+        __visited__: {
+          "!doc": "当前访问过的楼层",
         },
-        "__leaveLoc__": {
-          "!doc": "每个楼层的离开位置,用于楼传平面塔模式"
+        __leaveLoc__: {
+          "!doc": "每个楼层的离开位置,用于楼传平面塔模式",
         },
-        "cannotMoveDirectly": {
+        cannotMoveDirectly: {
           "!type": "bool",
-          "!doc": "当前是否全局不可瞬移"
+          "!doc": "当前是否全局不可瞬移",
         },
       },
-      "hero": {
+      hero: {
         "!doc": "勇士当前属性",
-        "image": {
+        image: {
           "!type": "string",
-          "!doc": "行走图"
+          "!doc": "行走图",
         },
-        "animate": {
+        animate: {
           "!type": "bool",
-          "!doc": "是否开启帧动画"
+          "!doc": "是否开启帧动画",
         },
-        "name": {
+        name: {
           "!type": "string",
-          "!doc": "勇士名"
+          "!doc": "勇士名",
         },
-        "lv": {
+        lv: {
           "!type": "number",
-          "!doc": "勇士等级"
+          "!doc": "勇士等级",
         },
-        "hpmax": {
+        hpmax: {
           "!type": "number",
-          "!doc": "勇士生命上限"
+          "!doc": "勇士生命上限",
         },
-        "hp": {
+        hp: {
           "!type": "number",
-          "!doc": "勇士当前生命值"
+          "!doc": "勇士当前生命值",
         },
-        "atk": {
+        atk: {
           "!type": "number",
-          "!doc": "勇士当前攻击力"
+          "!doc": "勇士当前攻击力",
         },
-        "def": {
+        def: {
           "!type": "number",
-          "!doc": "勇士当前防御力"
+          "!doc": "勇士当前防御力",
         },
-        "manamax": {
+        manamax: {
           "!type": "number",
-          "!doc": "勇士当前魔力上限,负数无效"
+          "!doc": "勇士当前魔力上限,负数无效",
         },
-        "mana": {
+        mana: {
           "!type": "number",
-          "!doc": "勇士当前魔力值"
+          "!doc": "勇士当前魔力值",
         },
-        "mdef": {
+        mdef: {
           "!type": "number",
-          "!doc": "勇士当前护盾值"
+          "!doc": "勇士当前护盾值",
         },
-        "money": {
+        money: {
           "!type": "number",
-          "!doc": "勇士当前金币"
+          "!doc": "勇士当前金币",
         },
-        "exp": {
+        exp: {
           "!type": "number",
-          "!doc": "勇士当前经验"
+          "!doc": "勇士当前经验",
         },
-        "equipment": {
+        equipment: {
           "!type": "[string]",
-          "!doc": "勇士当前装备"
+          "!doc": "勇士当前装备",
         },
-        "items": {
+        items: {
           "!doc": "勇士当前道具",
-          "constants": {
-            "!doc": "永久道具"
+          constants: {
+            "!doc": "永久道具",
           },
-          "tools": {
+          tools: {
             "!doc": "消耗道具",
-            "yellowKey": {
+            yellowKey: {
               "!type": "number",
-              "!doc": "黄钥匙个数"
+              "!doc": "黄钥匙个数",
             },
-            "blueKey": {
+            blueKey: {
               "!type": "number",
-              "!doc": "蓝钥匙个数"
+              "!doc": "蓝钥匙个数",
             },
-            "redKey": {
+            redKey: {
               "!type": "number",
-              "!doc": "红钥匙个数"
+              "!doc": "红钥匙个数",
             },
-            "greenKey": {
+            greenKey: {
               "!type": "number",
-              "!doc": "绿钥匙个数"
+              "!doc": "绿钥匙个数",
             },
-            "steelKey": {
+            steelKey: {
               "!type": "number",
-              "!doc": "铁门钥匙个数"
+              "!doc": "铁门钥匙个数",
             },
           },
-          "equips": {
-            "!doc": "在背包中未装备上的装备"
+          equips: {
+            "!doc": "在背包中未装备上的装备",
           },
         },
-        "loc": {
+        loc: {
           "!doc": "勇士当前坐标和朝向",
-          "x": "number",
-          "y": "number",
-          "direction": {
+          x: "number",
+          y: "number",
+          direction: {
             "!doc": "朝向,只能为 up,down,left,right 之一",
-            "!type": "string"
+            "!type": "string",
           },
         },
-        "flags": {
+        flags: {
           "!type": "flag",
-          "!doc": "当前游戏中用到的变量"
+          "!doc": "当前游戏中用到的变量",
         },
-        "followers": {
+        followers: {
           "!type": "[?]",
-          "!doc": "跟随者信息"
+          "!doc": "跟随者信息",
         },
-        "steps": {
+        steps: {
           "!type": "number",
-          "!doc": "当前步数"
-        }
+          "!doc": "当前步数",
+        },
       },
-      "block": {
+      block: {
         "!doc": "地图图块信息",
-        "x": {
+        x: {
           "!type": "number",
-          "!doc": "图块的x坐标"
+          "!doc": "图块的x坐标",
         },
-        "y": {
+        y: {
           "!type": "number",
-          "!doc": "图块的y坐标"
+          "!doc": "图块的y坐标",
         },
-        "id": {
+        id: {
           "!type": "number",
-          "!doc": "图块的数字"
+          "!doc": "图块的数字",
         },
-        "event": {
+        event: {
           "!doc": "图块上的事件信息",
-          "id": {
+          id: {
             "!type": "string",
-            "!doc": "图块的ID"
+            "!doc": "图块的ID",
           },
-          "cls": {
+          cls: {
             "!type": "string",
-            "!doc": "图块的类别,一般为所在图片名去掉后缀"
+            "!doc": "图块的类别,一般为所在图片名去掉后缀",
           },
-          "disabled": {
+          disabled: {
             "!type": "bool",
-            "!doc": "启用状态"
-          }
-        }
+            "!doc": "启用状态",
+          },
+        },
       },
-      "blockInfo": {
+      blockInfo: {
         "!doc": "图块的更多信息",
-        "animate": {
+        animate: {
           "!type": "number",
-          "!doc": "动画帧数"
+          "!doc": "动画帧数",
         },
-        "cls": {
+        cls: {
           "!type": "string",
-          "!doc": "图块类别"
+          "!doc": "图块类别",
         },
-        "faceIds": {
+        faceIds: {
           "!doc": "行走图朝向",
-          "up": "string",
-          "down": "string",
-          "left": "string",
-          "right": "string"
+          up: "string",
+          down: "string",
+          left: "string",
+          right: "string",
         },
-        "height": {
+        height: {
           "!type": "number",
-          "!doc": "图块高度"
+          "!doc": "图块高度",
         },
-        "id": {
+        id: {
           "!type": "string",
-          "!doc": "图块ID"
+          "!doc": "图块ID",
         },
-        "image": {
+        image: {
           "!type": "image",
-          "!doc": "图块所在的图片"
+          "!doc": "图块所在的图片",
         },
-        "name": {
+        name: {
           "!type": "string",
-          "!doc": "图块名称"
+          "!doc": "图块名称",
         },
-        "number": {
+        number: {
           "!type": "number",
-          "!doc": "图块使用的数字"
+          "!doc": "图块使用的数字",
         },
-        "posX": {
+        posX: {
           "!type": "number",
-          "!doc": "图块在图片上的横坐标"
+          "!doc": "图块在图片上的横坐标",
         },
-        "posY": {
+        posY: {
           "!type": "number",
-          "!doc": "图块在图片上的纵坐标"
+          "!doc": "图块在图片上的纵坐标",
         },
       },
-      "enemy": {
+      enemy: {
         "!doc": "怪物信息",
-        "id": {
+        id: {
           "!type": "string",
-          "!doc": "怪物ID"
+          "!doc": "怪物ID",
         },
-        "name": {
+        name: {
           "!type": "string",
-          "!doc": "怪物名称"
+          "!doc": "怪物名称",
         },
-        "displayIdInBook": {
+        displayIdInBook: {
           "!type": "string",
-          "!doc": "在怪物手册映射ID"
+          "!doc": "在怪物手册映射ID",
         },
-        "hp": {
+        hp: {
           "!type": "number",
-          "!doc": "怪物生命值"
+          "!doc": "怪物生命值",
         },
-        "atk": {
+        atk: {
           "!type": "number",
-          "!doc": "怪物攻击"
+          "!doc": "怪物攻击",
         },
-        "def": {
+        def: {
           "!type": "number",
-          "!doc": "怪物防御"
+          "!doc": "怪物防御",
         },
-        "money": {
+        money: {
           "!type": "number",
-          "!doc": "怪物金币"
+          "!doc": "怪物金币",
         },
-        "exp": {
+        exp: {
           "!type": "number",
-          "!doc": "怪物经验"
+          "!doc": "怪物经验",
         },
-        "special": {
+        special: {
           "!type": "[number]",
-          "!doc": "怪物特殊属性"
+          "!doc": "怪物特殊属性",
         },
-        "point": {
+        point: {
           "!type": "number",
-          "!doc": "怪物加点"
+          "!doc": "怪物加点",
         },
-        "value": {
+        value: {
           "!type": "number",
-          "!doc": "怪物特殊属性值:阻激夹域伤害值;吸血比例;光环增加生命比例"
+          "!doc": "怪物特殊属性值:阻激夹域伤害值;吸血比例;光环增加生命比例",
         },
-        "zoneSquare": {
+        zoneSquare: {
           "!type": "bool",
-          "!doc": "领域怪是否九宫格伤害;区域光环是否九宫格范围"
+          "!doc": "领域怪是否九宫格伤害;区域光环是否九宫格范围",
         },
-        "range": {
+        range: {
           "!type": "number",
-          "!doc": "领域伤害的范围;区域光环范围"
+          "!doc": "领域伤害的范围;区域光环范围",
         },
-        "notBomb": {
+        notBomb: {
           "!type": "bool",
-          "!doc": "怪物不可炸"
+          "!doc": "怪物不可炸",
         },
-        "n": {
+        n: {
           "!type": "number",
-          "!doc": "多连击的连击数;净化比例"
+          "!doc": "多连击的连击数;净化比例",
         },
-        "add": {
+        add: {
           "!type": "bool",
-          "!doc": "吸血是否加到自身;光环是否叠加"
+          "!doc": "吸血是否加到自身;光环是否叠加",
         },
-        "atkValue": {
+        atkValue: {
           "!type": "number",
-          "!doc": "反击比例;退化扣除攻击;光环增加攻击;"
+          "!doc": "反击比例;退化扣除攻击;光环增加攻击;",
         },
-        "defValue": {
+        defValue: {
           "!type": "number",
-          "!doc": "破甲比例;退化扣除防御;光环增加防御"
+          "!doc": "破甲比例;退化扣除防御;光环增加防御",
         },
-        "damage": {
+        damage: {
           "!type": "number",
-          "!doc": "固伤值"
+          "!doc": "固伤值",
         },
       },
-      "item": {
+      item: {
         "!doc": "道具信息",
-        "id": {
+        id: {
           "!type": "string",
-          "!doc": "道具ID"
+          "!doc": "道具ID",
         },
-        "cls": {
+        cls: {
           "!type": "string",
-          "!doc": "道具类型"
+          "!doc": "道具类型",
         },
-        "name": {
+        name: {
           "!type": "string",
-          "!doc": "道具名称"
+          "!doc": "道具名称",
         },
-        "text": {
+        text: {
           "!type": "string",
-          "!doc": "道具描述"
+          "!doc": "道具描述",
         },
-        "hideInToolbox": {
+        hideInToolbox: {
           "!type": "bool",
-          "!doc": "不显示在道具栏"
+          "!doc": "不显示在道具栏",
         },
-        "equip": {
+        equip: {
           "!doc": "装备属性",
-          "type": {
+          type: {
             "!type": "number|string",
-            "!doc": "装备类型"
+            "!doc": "装备类型",
           },
-          "animate": {
+          animate: {
             "!type": "string",
-            "!doc": "装备动画"
+            "!doc": "装备动画",
           },
-          "value": {
-            "!doc": "数值加成"
+          value: {
+            "!doc": "数值加成",
+          },
+          percentage: {
+            "!doc": "比例加成",
           },
-          "percentage": {
-            "!doc": "比例加成"
-          }
         },
-        "hideInReplay": {
+        hideInReplay: {
           "!type": "bool",
-          "!doc": "回放不绘制道具栏"
+          "!doc": "回放不绘制道具栏",
         },
       },
-      "floor": {
+      floor: {
         "!doc": "楼层信息",
-        "floorId": {
+        floorId: {
           "!type": "string",
-          "!doc": "楼层ID"
+          "!doc": "楼层ID",
         },
-        "title": {
+        title: {
           "!type": "string",
-          "!doc": "楼层中文名"
+          "!doc": "楼层中文名",
         },
-        "name": {
+        name: {
           "!type": "string",
-          "!doc": "状态栏显示值"
+          "!doc": "状态栏显示值",
         },
-        "width": {
+        width: {
           "!type": "number",
-          "!doc": "地图宽"
+          "!doc": "地图宽",
         },
-        "height": {
+        height: {
           "!type": "number",
-          "!doc": "地图高"
+          "!doc": "地图高",
         },
-        "canFlyTo": {
+        canFlyTo: {
           "!type": "bool",
-          "!doc": "该楼是否可以楼传飞到"
+          "!doc": "该楼是否可以楼传飞到",
         },
-        "canFlyFrom": {
+        canFlyFrom: {
           "!type": "bool",
-          "!doc": "该楼是否可以楼传飞出"
+          "!doc": "该楼是否可以楼传飞出",
         },
-        "canUseQuickShop": {
+        canUseQuickShop: {
           "!type": "bool",
-          "!doc": "该楼是否可快捷商店"
+          "!doc": "该楼是否可快捷商店",
         },
-        "cannotViewMap": {
+        cannotViewMap: {
           "!type": "bool",
-          "!doc": "该层是否不允许被浏览地图看到,也不统计"
+          "!doc": "该层是否不允许被浏览地图看到,也不统计",
         },
-        "cannotMoveDirectly": {
+        cannotMoveDirectly: {
           "!type": "bool",
-          "!doc": "该层是否不允许瞬间移动"
+          "!doc": "该层是否不允许瞬间移动",
         },
-        "upFloor": {
+        upFloor: {
           "!type": "[number]",
-          "!doc": "上楼点"
+          "!doc": "上楼点",
         },
-        "downFloor": {
+        downFloor: {
           "!type": "[number]",
-          "!doc": "下楼点"
+          "!doc": "下楼点",
         },
-        "flyPoint": {
+        flyPoint: {
           "!type": "[number]",
-          "!doc": "楼传落点"
+          "!doc": "楼传落点",
         },
-        "color": {
-          "!doc": "楼层色调"
+        color: {
+          "!doc": "楼层色调",
         },
-        "weather": {
-          "!doc": "楼层天气"
+        weather: {
+          "!doc": "楼层天气",
         },
-        "bgm": {
+        bgm: {
           "!type": "string",
-          "!doc": "楼层背景音乐"
+          "!doc": "楼层背景音乐",
         },
-        "ratio": {
+        ratio: {
           "!type": "number",
-          "!doc": "宝石/血瓶效果"
+          "!doc": "宝石/血瓶效果",
         },
-        "map": {
+        map: {
           "!type": "[[number]]",
-          "!doc": "地图数据"
+          "!doc": "地图数据",
         },
-        "blocks": {
+        blocks: {
           "!type": "[block]",
-          "!doc": "本层图块信息"
-        }
+          "!doc": "本层图块信息",
+        },
       },
-      "animate": {
+      animate: {
         "!doc": "动画信息",
-        "se": {
+        se: {
           "!type": "string",
-          "!doc": "动画音效"
-        }
-      }
+          "!doc": "动画音效",
+        },
+      },
     },
-    "core": {
+    core: {
       "!doc": "核心游戏控制",
-      "__SIZE__": {
+      __SIZE__: {
         "!type": "number",
-        "!doc": "窗口宽度,为13或15"
+        "!doc": "窗口宽度,为13或15",
       },
-      "__PIXELS__": {
+      __PIXELS__: {
         "!type": "number",
-        "!doc": "窗口像素宽度,为416或480"
+        "!doc": "窗口像素宽度,为416或480",
       },
-      "__HALF_SIZE__": {
+      __HALF_SIZE__: {
         "!type": "number",
-        "!doc": "窗口宽度的一半,为6或7"
+        "!doc": "窗口宽度的一半,为6或7",
       },
-      "floorIds": {
+      floorIds: {
         "!type": "[string]",
-        "!doc": "全部楼层ID列表"
+        "!doc": "全部楼层ID列表",
       },
-      "floors": {
-        "!doc": "全部楼层信息"
+      floors: {
+        "!doc": "全部楼层信息",
       },
-      "floorPartitions": {
+      floorPartitions: {
         "!type": "[[string]]",
-        "!doc": "楼层分区信息"
+        "!doc": "楼层分区信息",
       },
-      "material": {
+      material: {
         "!doc": "游戏所用到的资源",
-        "animates": {
-          "!doc": "注册的动画"
+        animates: {
+          "!doc": "注册的动画",
         },
-        "images": {
-          "!doc": "注册的图片"
+        images: {
+          "!doc": "注册的图片",
         },
-        "bgms": {
-          "!doc": "注册的背景音乐"
+        bgms: {
+          "!doc": "注册的背景音乐",
         },
-        "sounds": {
-          "!doc": "注册的音效"
+        sounds: {
+          "!doc": "注册的音效",
         },
-        "enemys": {
+        enemys: {
           "!doc": "怪物定义",
         },
-        "items": {
-          "!doc": "道具定义"
-        }
+        items: {
+          "!doc": "道具定义",
+        },
       },
-      "timeout": {
-        "!doc": "当前异步事件句柄"
+      timeout: {
+        "!doc": "当前异步事件句柄",
       },
-      "interval": {
-        "!doc": "当前异步事件延时"
+      interval: {
+        "!doc": "当前异步事件延时",
       },
-      "animateFrame": {
-        "!doc": "当前各个帧动画"
+      animateFrame: {
+        "!doc": "当前各个帧动画",
       },
-      "musicStatus": {
+      musicStatus: {
         "!doc": "音乐音效状态",
-        "bgmStatus": {
+        bgmStatus: {
           "!type": "bool",
-          "!doc": "是否播放BGM"
+          "!doc": "是否播放BGM",
         },
-        "soundStatus": {
+        soundStatus: {
           "!type": "bool",
-          "!doc": "是否播放SE"
+          "!doc": "是否播放SE",
         },
-        "playingBgm": {
+        playingBgm: {
           "!type": "string",
-          "!doc": "正在播放的bgm"
+          "!doc": "正在播放的bgm",
         },
-        "lastBgm": {
+        lastBgm: {
           "!type": "string",
-          "!doc": "上次播放的bgm"
+          "!doc": "上次播放的bgm",
         },
-        "playingSounds": {
-          "!doc": "正在播放的SE"
+        playingSounds: {
+          "!doc": "正在播放的SE",
         },
-        "volume": {
+        volume: {
           "!type": "number",
-          "!doc": "当前bgm音量"
-        }
+          "!doc": "当前bgm音量",
+        },
       },
-      "platform": {
+      platform: {
         "!doc": "平台信息",
-        "isPC": "bool",
-        "isAndroid": "bool",
-        "isIOS": "bool",
+        isPC: "bool",
+        isAndroid: "bool",
+        isIOS: "bool",
       },
-      "domStyle": {
+      domStyle: {
         "!doc": "界面样式",
-        "scale": {
+        scale: {
           "!type": "number",
           "!doc": "当前界面放缩比例",
         },
-        "ratio": {
+        ratio: {
           "!type": "number",
-          "!doc": "高清UI放缩比例"
+          "!doc": "高清UI放缩比例",
         },
-        "hdCanvas": {
+        hdCanvas: {
           "!type": "[string]",
-          "!doc": "高清UI的系统画布"
+          "!doc": "高清UI的系统画布",
         },
-        "availableScale": {
+        availableScale: {
           "!type": "[number]",
-          "!doc": "当前界面支持的放缩比例"
+          "!doc": "当前界面支持的放缩比例",
         },
-        "isVertical": {
+        isVertical: {
           "!type": "bool",
-          "!doc": "当前是否是竖屏"
+          "!doc": "当前是否是竖屏",
         },
-        "showStatusBar": {
+        showStatusBar: {
           "!type": "bool",
-          "!doc": "当前是否显示状态栏"
+          "!doc": "当前是否显示状态栏",
         },
-        "toolbarBtn": {
+        toolbarBtn: {
           "!type": "bool",
-          "!doc": "当前工具栏是否是1-8的按钮"
+          "!doc": "当前工具栏是否是1-8的按钮",
         },
       },
-      "bigmap": {
+      bigmap: {
         "!doc": "大地图信息",
-        "canvas": {
+        canvas: {
           "!type": "[string]",
-          "!doc": "大地图的画布"
+          "!doc": "大地图的画布",
         },
-        "width": {
+        width: {
           "!type": "number",
-          "!doc": "大地图高度"
+          "!doc": "大地图高度",
         },
-        "height": {
+        height: {
           "!type": "number",
-          "!doc": "大地图宽度"
+          "!doc": "大地图宽度",
         },
-        "offsetX": {
+        offsetX: {
           "!type": "number",
-          "!doc": "大地图视角横向偏移量"
+          "!doc": "大地图视角横向偏移量",
         },
-        "offsetY": {
+        offsetY: {
           "!type": "number",
-          "!doc": "大地图视角纵向偏移量"
+          "!doc": "大地图视角纵向偏移量",
         },
-        "posX": {
+        posX: {
           "!type": "number",
-          "!doc": "大地图视角横向基准格"
+          "!doc": "大地图视角横向基准格",
         },
-        "posY": {
+        posY: {
           "!type": "number",
-          "!doc": "大地图视角纵向基准格"
+          "!doc": "大地图视角纵向基准格",
         },
-        "v2": {
+        v2: {
           "!type": "bool",
-          "!doc": "是否是新版大地图绘制方式"
+          "!doc": "是否是新版大地图绘制方式",
         },
-        "threshold": {
+        threshold: {
           "!type": "number",
-          "!doc": "新版大地图绘制方式的分界线"
+          "!doc": "新版大地图绘制方式的分界线",
         },
-        "extend": {
+        extend: {
           "!type": "number",
-          "!doc": "新版大地图模式下向每一侧额外计算的数量"
+          "!doc": "新版大地图模式下向每一侧额外计算的数量",
         },
-        "scale": {
+        scale: {
           "!type": "number",
-          "!doc": "缩略图的比例放缩"
+          "!doc": "缩略图的比例放缩",
         },
-        "tempCanvas": {
+        tempCanvas: {
           "!type": "CanvasRenderingContext2D",
-          "!doc": "临时画布"
-        }
+          "!doc": "临时画布",
+        },
       },
-      "saves": {
-        "!doc": "当前存档信息"
+      saves: {
+        "!doc": "当前存档信息",
       },
-      "dymCanvas": {
-        "!doc": "各个自定义画布"
+      dymCanvas: {
+        "!doc": "各个自定义画布",
       },
-      "statusBar": {
-        "!doc": "状态栏信息"
+      statusBar: {
+        "!doc": "状态栏信息",
       },
-      "canvas": {
-        "!doc": "系统画布"
+      canvas: {
+        "!doc": "系统画布",
       },
-      "flags": {
-        "!doc": "系统开关"
+      flags: {
+        "!doc": "系统开关",
       },
-      "values": {
-        "!doc": "全局数值,如毒衰效果"
+      values: {
+        "!doc": "全局数值,如毒衰效果",
       },
-      "firstData": {
-        "!doc": "初始属性,如出生点"
+      firstData: {
+        "!doc": "初始属性,如出生点",
       },
-      "status": {
+      status: {
         "!doc": "状态信息",
-        "hero": {
+        hero: {
           "!type": "hero",
-          "!doc": "勇士信息"
+          "!doc": "勇士信息",
         },
-        "automaticRoute": {
-          "!doc": "自动寻路信息"
+        automaticRoute: {
+          "!doc": "自动寻路信息",
         },
-        "bgmaps": {
-          "!doc": "各地图背景层"
+        bgmaps: {
+          "!doc": "各地图背景层",
         },
-        "fgmaps": {
-          "!doc": "各地图前景层"
+        fgmaps: {
+          "!doc": "各地图前景层",
         },
-        "mapBlockObjs": {
-          "!doc": "以<位置,block>存放的各地图图块信息"
+        mapBlockObjs: {
+          "!doc": "以<位置,block>存放的各地图图块信息",
         },
-        "boxAnimateObjs": {
-          "!doc": "(手册和剧情文本的)帧动画对象"
+        boxAnimateObjs: {
+          "!doc": "(手册和剧情文本的)帧动画对象",
         },
-        "checkBlock": {
+        checkBlock: {
           "!doc": "阻激夹域捕捉信息",
-          "damage": {
-            "!doc": "每个点的伤害信息"
+          damage: {
+            "!doc": "每个点的伤害信息",
           },
-          "type": {
-            "!doc": "每个点的伤害类型"
+          type: {
+            "!doc": "每个点的伤害类型",
           },
-          "repluse": {
-            "!doc": "每个点的阻击信息"
+          repluse: {
+            "!doc": "每个点的阻击信息",
           },
-          "ambush": {
-            "!doc": "每个点的捕捉信息"
+          ambush: {
+            "!doc": "每个点的捕捉信息",
           },
-          "needCache": {
+          needCache: {
             "!type": "bool",
-            "!doc": "该楼层是否需要计算缓存"
+            "!doc": "该楼层是否需要计算缓存",
           },
-          "cache": {
-            "!doc": "每个点的光环缓存"
+          cache: {
+            "!doc": "每个点的光环缓存",
           },
         },
-        "damage": {
+        damage: {
           "!doc": "每个点的显伤信息",
         },
-        "ctrlDown": {
+        ctrlDown: {
           "!type": "bool",
-          "!doc": "Ctrl键是否被按下"
+          "!doc": "Ctrl键是否被按下",
         },
-        "curtainColor": {
-          "!doc": "当前画面色调"
+        curtainColor: {
+          "!doc": "当前画面色调",
         },
-        "event": {
+        event: {
           "!doc": "当前事件",
-          "data": {
-            "!doc": "事件信息,如坐标等"
+          data: {
+            "!doc": "事件信息,如坐标等",
           },
-          "id": {
+          id: {
             "!type": "string",
-            "!doc": "事件类型,如选择项/确认框"
+            "!doc": "事件类型,如选择项/确认框",
           },
-          "interval": {
+          interval: {
             "!type": "number",
-            "!doc": "打字机效果的定时器"
+            "!doc": "打字机效果的定时器",
           },
-          "selection": {
+          selection: {
             "!type": "number",
-            "!doc": "选择项和确认框的当前选中项"
+            "!doc": "选择项和确认框的当前选中项",
+          },
+          ui: {
+            "!doc": "当前事件的界面信息,如楼传/手册/SL",
           },
-          "ui": {
-            "!doc": "当前事件的界面信息,如楼传/手册/SL"
-          }
         },
-        "floorAnimateObjs": {
-          "!doc": "楼层贴图的帧动画"
+        floorAnimateObjs: {
+          "!doc": "楼层贴图的帧动画",
         },
-        "floorId": {
+        floorId: {
           "!type": "string",
-          "!doc": "当前楼层ID"
+          "!doc": "当前楼层ID",
         },
-        "gameOver": {
+        gameOver: {
           "!type": "bool",
-          "!doc": "游戏是否已结束"
+          "!doc": "游戏是否已结束",
         },
-        "globalAnimateObjs": {
-          "!doc": "各全局动画"
+        globalAnimateObjs: {
+          "!doc": "各全局动画",
         },
-        "globalAnimateStatus": {
+        globalAnimateStatus: {
           "!type": "number",
-          "!doc": "全局动画的帧状态"
+          "!doc": "全局动画的帧状态",
         },
-        "globalAttribute": {
-          "!doc": "全局css属性"
+        globalAttribute: {
+          "!doc": "全局css属性",
         },
-        "hard": {
+        hard: {
           "!type": "string",
-          "!doc": "状态栏一角的难度名"
+          "!doc": "状态栏一角的难度名",
         },
-        "downTime": {
+        downTime: {
           "!type": "number",
-          "!doc": "方向键已按下的时间"
+          "!doc": "方向键已按下的时间",
         },
-        "heroCenter": {
+        heroCenter: {
           "!doc": "勇士中心像素坐标",
-          "px": {
+          px: {
             "!type": "number",
-            "!doc": "勇士中心的横坐标"
+            "!doc": "勇士中心的横坐标",
           },
-          "py": {
+          py: {
             "!type": "number",
-            "!doc": "勇士中心的纵坐标"
+            "!doc": "勇士中心的纵坐标",
           },
         },
-        "heroMoving": {
+        heroMoving: {
           "!type": "number",
-          "!doc": "勇士行走的状态值"
+          "!doc": "勇士行走的状态值",
         },
-        "heroStop": {
+        heroStop: {
           "!type": "bool",
-          "!doc": "勇士是否已停下"
+          "!doc": "勇士是否已停下",
         },
-        "holdingKeys": {
+        holdingKeys: {
           "!type": "[number]",
-          "!doc": "当前按下的键"
+          "!doc": "当前按下的键",
         },
-        "id2number": {
-          "!doc": "图块ID到数字的对应关系"
+        id2number: {
+          "!doc": "图块ID到数字的对应关系",
         },
-        "lockControl": {
+        lockControl: {
           "!type": "bool",
-          "!doc": "当前是否是锁定操作状态"
+          "!doc": "当前是否是锁定操作状态",
         },
-        "maps": {
-          "!doc": "当前各地图信息"
+        maps: {
+          "!doc": "当前各地图信息",
         },
-        "number2Block": {
-          "!doc": "数字到图块对象的对应关系"
+        number2Block: {
+          "!doc": "数字到图块对象的对应关系",
         },
-        "played": {
+        played: {
           "!type": "bool",
-          "!doc": "当前是否游戏中(不包括标题画面和录像回放)"
+          "!doc": "当前是否游戏中(不包括标题画面和录像回放)",
         },
-        "replay": {
+        replay: {
           "!doc": "当前录像回放信息",
-          "animate": {
+          animate: {
             "!type": "bool",
-            "!doc": "回放是否正处于动画中"
+            "!doc": "回放是否正处于动画中",
           },
-          "pausing": {
+          pausing: {
             "!type": "bool",
-            "!doc": "回放是否暂停中"
+            "!doc": "回放是否暂停中",
           },
-          "replaying": {
+          replaying: {
             "!type": "bool",
-            "!doc": "当前是否回放中"
+            "!doc": "当前是否回放中",
           },
-          "save": {
+          save: {
             "!type": "[]",
-            "!doc": "录像中的存档"
+            "!doc": "录像中的存档",
           },
-          "speed": {
+          speed: {
             "!type": "number",
-            "!doc": "回放速度"
+            "!doc": "回放速度",
           },
-          "steps": {
+          steps: {
             "!type": "number",
-            "!doc": "回放步数"
+            "!doc": "回放步数",
           },
-          "toReplay": {
+          toReplay: {
             "!type": "[string]",
-            "!doc": "待回放的列表"
+            "!doc": "待回放的列表",
           },
-          "totalList": {
+          totalList: {
             "!type": "[string]",
-            "!doc": "回放总列表"
-          }
+            "!doc": "回放总列表",
+          },
         },
-        "route": {
+        route: {
           "!type": "[string]",
-          "!doc": "当前录像内容"
+          "!doc": "当前录像内容",
         },
-        "shops": {
-          "!doc": "全局商店列表"
+        shops: {
+          "!doc": "全局商店列表",
         },
-        "textAttribute": {
-          "!doc": "当前剧情文本属性"
+        textAttribute: {
+          "!doc": "当前剧情文本属性",
         },
-        "thisMap": {
+        thisMap: {
           "!type": "floor",
-          "!doc": "当前地图信息"
-        }
+          "!doc": "当前地图信息",
+        },
       },
-      "init": {
+      init: {
         "!doc": "初始化core",
-        "!type": "fn(coreData: ?, callback: fn())"
+        "!type": "fn(coreData: ?, callback: fn())",
       },
-      "doFunc": {
+      doFunc: {
         "!doc": "执行一个函数;如果函数名是字符串则转发到插件中",
-        "!type": "fn(func: name|fn(), _this?: ?)"
+        "!type": "fn(func: name|fn(), _this?: ?)",
       },
-      "control": {
-        "!doc": "负责整个游戏的核心控制系统,分为如下几个部分:<br/>- requestAnimationFrame相关<br/>- 标题界面,开始和重新开始游戏<br/>- 自动寻路和人物行走相关<br/>- 画布、位置、阻激夹域、显伤等相关<br/>- 录像的回放相关<br/>- 存读档,自动存档,同步存档等相关<br/>- 人物属性和状态、位置、变量等相关<br/>- 天气、色调、音乐和音效的播放<br/>- 状态栏和工具栏相关<br/>- 界面resize相关",
-        "showStatusBar": {
+      control: {
+        "!doc":
+          "负责整个游戏的核心控制系统,分为如下几个部分:<br/>- requestAnimationFrame相关<br/>- 标题界面,开始和重新开始游戏<br/>- 自动寻路和人物行走相关<br/>- 画布、位置、阻激夹域、显伤等相关<br/>- 录像的回放相关<br/>- 存读档,自动存档,同步存档等相关<br/>- 人物属性和状态、位置、变量等相关<br/>- 天气、色调、音乐和音效的播放<br/>- 状态栏和工具栏相关<br/>- 界面resize相关",
+        showStatusBar: {
           "!doc": "显示状态栏",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "startReplay": {
+        startReplay: {
           "!doc": "开始播放录像",
-          "!type": "fn(list: [string])"
+          "!type": "fn(list: [string])",
         },
-        "triggerReplay": {
+        triggerReplay: {
           "!doc": "播放或暂停录像回放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "screenFlash": {
-          "!doc": "画面闪烁<br/>例如:core.screenFlash([255, 0, 0, 1], 3); // 红屏一闪而过<br/>color: 一行三列(第四列视为1)或一行四列(第四列若大于1则会被视为1,第四列若填负数则会被视为0)的颜色数组,必填<br/>time: 单次闪烁时长,实际闪烁效果为先花其三分之一的时间渐变到目标色调,再花剩余三分之二的时间渐变回去<br/>times: 闪烁的总次数,不填或填0都视为1<br/>moveMode: 渐变方式<br/>callback: 闪烁全部完毕后的回调函数,可选",
-          "!type": "fn(color: [number], time: number, times?: number, moveMode?: string, callback?: fn())"
+        screenFlash: {
+          "!doc":
+            "画面闪烁<br/>例如:core.screenFlash([255, 0, 0, 1], 3); // 红屏一闪而过<br/>color: 一行三列(第四列视为1)或一行四列(第四列若大于1则会被视为1,第四列若填负数则会被视为0)的颜色数组,必填<br/>time: 单次闪烁时长,实际闪烁效果为先花其三分之一的时间渐变到目标色调,再花剩余三分之二的时间渐变回去<br/>times: 闪烁的总次数,不填或填0都视为1<br/>moveMode: 渐变方式<br/>callback: 闪烁全部完毕后的回调函数,可选",
+          "!type":
+            "fn(color: [number], time: number, times?: number, moveMode?: string, callback?: fn())",
         },
-        "setCurtain": {
-          "!doc": "更改画面色调,不计入存档。如需长期生效请使用core.events._action_setCurtain()函数<br/>例如:core.setCurtain(); // 恢复画面色调,用时四分之三秒<br/>color: 一行三列(第四列视为1)或一行四列(第四列若大于1则会被视为1,第四列若为负数则会被视为0)的颜色数组,不填视为[0, 0, 0, 0]<br/>time: 渐变时间,单位为毫秒。不填视为750ms,负数视为0(无渐变,立即更改)<br/>moveMode: 渐变方式<br/>callback: 更改完毕后的回调函数,可选。事件流中常取core.doAction",
-          "!type": "fn(color?: [number], time?: number, moveMode?: string, callback?: fn())"
+        setCurtain: {
+          "!doc":
+            "更改画面色调,不计入存档。如需长期生效请使用core.events._action_setCurtain()函数<br/>例如:core.setCurtain(); // 恢复画面色调,用时四分之三秒<br/>color: 一行三列(第四列视为1)或一行四列(第四列若大于1则会被视为1,第四列若为负数则会被视为0)的颜色数组,不填视为[0, 0, 0, 0]<br/>time: 渐变时间,单位为毫秒。不填视为750ms,负数视为0(无渐变,立即更改)<br/>moveMode: 渐变方式<br/>callback: 更改完毕后的回调函数,可选。事件流中常取core.doAction",
+          "!type":
+            "fn(color?: [number], time?: number, moveMode?: string, callback?: fn())",
         },
-        "updateDamage": {
-          "!doc": "注意!请勿使用该函数!请使用core.updateStatusBar()代替!!重算并绘制地图显伤<br/>例如:core.updateDamage(); // 更新当前地图的显伤,绘制在显伤层(废话)<br/>floorId: 地图id,不填视为当前地图。预览地图时填写<br/>ctx: 绘制到的画布,如果填写了就会画在该画布而不是显伤层",
-          "!type": "fn(floorId?: string, ctx?: string|CanvasRenderingContext2D)"
+        updateDamage: {
+          "!doc":
+            "注意!请勿使用该函数!请使用core.updateStatusBar()代替!!重算并绘制地图显伤<br/>例如:core.updateDamage(); // 更新当前地图的显伤,绘制在显伤层(废话)<br/>floorId: 地图id,不填视为当前地图。预览地图时填写<br/>ctx: 绘制到的画布,如果填写了就会画在该画布而不是显伤层",
+          "!type":
+            "fn(floorId?: string, ctx?: string|CanvasRenderingContext2D)",
         },
-        "drawDamage": {
+        drawDamage: {
           "!doc": "仅绘制地图显伤",
-          "!type": "fn(string|CanvasRenderingContext2D)"
+          "!type": "fn(string|CanvasRenderingContext2D)",
         },
-        "nextX": {
-          "!doc": "获取主角面前第n格的横坐标<br/>例如:core.closeDoor(core.nextX(), core.nextY(), 'yellowDoor', core.turnHero); // 在主角面前关上一扇黄门,然后主角顺时针旋转90°<br/>n: 目标格与主角的距离,面前为正数,背后为负数,脚下为0,不填视为1",
-          "!type": "fn(n?: number) -> number"
+        nextX: {
+          "!doc":
+            "获取主角面前第n格的横坐标<br/>例如:core.closeDoor(core.nextX(), core.nextY(), 'yellowDoor', core.turnHero); // 在主角面前关上一扇黄门,然后主角顺时针旋转90°<br/>n: 目标格与主角的距离,面前为正数,背后为负数,脚下为0,不填视为1",
+          "!type": "fn(n?: number) -> number",
         },
-        "nextY": {
-          "!doc": "获取主角面前第n格的纵坐标<br/>例如:core.jumpHero(core.nextX(2), core.nextY(2)); // 主角向前跃过一格,即跳跃靴道具的使用效果<br/>n: 目标格与主角的距离,面前为正数,背后为负数,脚下为0,不填视为1",
-          "!type": "fn(n?: number) -> number"
+        nextY: {
+          "!doc":
+            "获取主角面前第n格的纵坐标<br/>例如:core.jumpHero(core.nextX(2), core.nextY(2)); // 主角向前跃过一格,即跳跃靴道具的使用效果<br/>n: 目标格与主角的距离,面前为正数,背后为负数,脚下为0,不填视为1",
+          "!type": "fn(n?: number) -> number",
         },
-        "clearContinueAutomaticRoute": {
+        clearContinueAutomaticRoute: {
           "!doc": "清空剩下的自动寻路列表",
-          "!type": "fn(callback?: fn())"
+          "!type": "fn(callback?: fn())",
         },
-        "updateViewport": {
+        updateViewport: {
           "!doc": "更新大地图的可见区域",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "getMappedName": {
+        getMappedName: {
           "!doc": "获得映射文件名",
-          "!type": "fn(name: string) -> string"
+          "!type": "fn(name: string) -> string",
         },
-        "addFlag": {
-          "!doc": "增减一个flag变量,等价于 core.setFlag(name, core.getFlag(name, 0) + value)<br/>例如:core.addFlag('hatred', 1); // 增加1点仇恨值<br/>name: 变量名,支持中文<br/>value: 变量的增量",
-          "!type": "fn(name: string, value: number)"
+        addFlag: {
+          "!doc":
+            "增减一个flag变量,等价于 core.setFlag(name, core.getFlag(name, 0) + value)<br/>例如:core.addFlag('hatred', 1); // 增加1点仇恨值<br/>name: 变量名,支持中文<br/>value: 变量的增量",
+          "!type": "fn(name: string, value: number)",
         },
-        "setFlag": {
-          "!doc": "设置一个flag变量<br/>例如:core.setFlag('poison', true); // 令主角中毒<br/>name: 变量名,支持中文<br/>value: 变量的新值,不填或填null视为删除",
-          "!type": "fn(name: string, value: ?)"
+        setFlag: {
+          "!doc":
+            "设置一个flag变量<br/>例如:core.setFlag('poison', true); // 令主角中毒<br/>name: 变量名,支持中文<br/>value: 变量的新值,不填或填null视为删除",
+          "!type": "fn(name: string, value: ?)",
         },
-        "playSound": {
-          "!doc": "播放一个音效<br/>sound: 音效名;可以使用文件别名。<br/>pitch: 播放的音调;可选,如果设置则为30-300之间的数值。<br/>callback: 可选,播放完毕后执行的回调函数。<br/>返回:一个数字,可用于core.stopSound的参数来只停止该音效。",
-          "!type": "fn(sound: string, pitch?: number, callback?: fn()) -> number"
+        playSound: {
+          "!doc":
+            "播放一个音效<br/>sound: 音效名;可以使用文件别名。<br/>pitch: 播放的音调;可选,如果设置则为30-300之间的数值。<br/>callback: 可选,播放完毕后执行的回调函数。<br/>返回:一个数字,可用于core.stopSound的参数来只停止该音效。",
+          "!type":
+            "fn(sound: string, pitch?: number, callback?: fn()) -> number",
         },
-        "stopSound": {
-          "!doc": "停止播放音效。如果未指定id则停止所有音效,否则只停止指定的音效。",
-          "!type": "fn(id?: number)"
+        stopSound: {
+          "!doc":
+            "停止播放音效。如果未指定id则停止所有音效,否则只停止指定的音效。",
+          "!type": "fn(id?: number)",
         },
-        "getPlayingSounds": {
-          "!doc": "获得当前正在播放的所有(指定)音效的id列表<br/>name: 音效名,可用别名;不填代表返回正在播放的全部音效<br/>返回值: 一个列表,每一项为一个正在播放的音效id;可用core.stopSound立刻停止播放",
-          "!type": "fn(name?: string) -> [number]"
+        getPlayingSounds: {
+          "!doc":
+            "获得当前正在播放的所有(指定)音效的id列表<br/>name: 音效名,可用别名;不填代表返回正在播放的全部音效<br/>返回值: 一个列表,每一项为一个正在播放的音效id;可用core.stopSound立刻停止播放",
+          "!type": "fn(name?: string) -> [number]",
         },
-        "addGameCanvasTranslate": {
+        addGameCanvasTranslate: {
           "!doc": "加减画布偏移",
-          "!type": "fn(x?: number, y?: number)"
+          "!type": "fn(x?: number, y?: number)",
         },
-        "addBuff": {
-          "!doc": "增减主角某个属性的百分比修正倍率,加减法叠加和抵消。等价于 core.setBuff(name, core.getBuff(name) + value)<br/>例如:core.addBuff('atk', -0.1); // 主角获得一层“攻击力减一成”的负面效果<br/>name: 属性的英文名,请注意只能用于数值类属性哦,否则随后的乘法会得到NaN<br/>value: 倍率的增量",
-          "!type": "fn(name: string, value: number)"
+        addBuff: {
+          "!doc":
+            "增减主角某个属性的百分比修正倍率,加减法叠加和抵消。等价于 core.setBuff(name, core.getBuff(name) + value)<br/>例如:core.addBuff('atk', -0.1); // 主角获得一层“攻击力减一成”的负面效果<br/>name: 属性的英文名,请注意只能用于数值类属性哦,否则随后的乘法会得到NaN<br/>value: 倍率的增量",
+          "!type": "fn(name: string, value: number)",
         },
-        "drawHero": {
-          "!doc": "绘制主角和跟随者并重置视野到以主角为中心<br/>例如:core.drawHero(); // 原地绘制主角的静止帧并重置视野野<br/>status: 只能为 stop, leftFoot 和 rightFoot,不填用stop。<br/>offset: 相对主角逻辑位置的偏移量,不填视为无偏移。<br/>frame: 绘制的第几帧",
-          "!type": "fn(status?: string, offset?: number, frame?: number)"
+        drawHero: {
+          "!doc":
+            "绘制主角和跟随者并重置视野到以主角为中心<br/>例如:core.drawHero(); // 原地绘制主角的静止帧并重置视野野<br/>status: 只能为 stop, leftFoot 和 rightFoot,不填用stop。<br/>offset: 相对主角逻辑位置的偏移量,不填视为无偏移。<br/>frame: 绘制的第几帧",
+          "!type": "fn(status?: string, offset?: number, frame?: number)",
         },
-        "pauseBgm": {
+        pauseBgm: {
           "!doc": "暂停背景音乐的播放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setBgmSpeed": {
-          "!doc": "设置背景音乐的播放速度和音调<br/>speed: 播放速度,必须为30-300中间的值。100为正常速度。<br/>usePitch: 是否同时改变音调(部分设备可能不支持)",
-          "!type": "fn(speed: number, usePitch?: bool)"
+        setBgmSpeed: {
+          "!doc":
+            "设置背景音乐的播放速度和音调<br/>speed: 播放速度,必须为30-300中间的值。100为正常速度。<br/>usePitch: 是否同时改变音调(部分设备可能不支持)",
+          "!type": "fn(speed: number, usePitch?: bool)",
         },
-        "setReplaySpeed": {
+        setReplaySpeed: {
           "!doc": "设置播放速度",
-          "!type": "fn(speed: number)"
+          "!type": "fn(speed: number)",
         },
-        "pauseReplay": {
+        pauseReplay: {
           "!doc": "暂停播放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "doSL": {
+        doSL: {
           "!doc": "实际进行存读档事件",
-          "!type": "fn(id?: string, type?: string)"
+          "!type": "fn(id?: string, type?: string)",
         },
-        "setStatus": {
-          "!doc": "设置主角的某个属性<br/>例如:core.setStatus('atk', 100); // 设置攻击力为100<br/>name: 属性的英文名,其中'x'、'y'和'direction'会被特殊处理为 core.setHeroLoc(name, value),其他的会直接对 core.status.hero[name] 赋值<br/>value: 属性的新值",
-          "!type": "fn(name: string, value: number)"
+        setStatus: {
+          "!doc":
+            "设置主角的某个属性<br/>例如:core.setStatus('atk', 100); // 设置攻击力为100<br/>name: 属性的英文名,其中'x'、'y'和'direction'会被特殊处理为 core.setHeroLoc(name, value),其他的会直接对 core.status.hero[name] 赋值<br/>value: 属性的新值",
+          "!type": "fn(name: string, value: number)",
         },
-        "setAutomaticRoute": {
-          "!doc": "半自动寻路,用于鼠标或手指拖动<br/>例如:core.setAutomaticRoute(0, 0, [{direction: \"right\", x: 4, y: 9}, {direction: \"right\", x: 5, y: 9}]);<br/>destX: 鼠标或手指的起拖点横坐标<br/>destY: 鼠标或手指的起拖点纵坐标<br/>stepPostfix: 拖动轨迹的数组表示,每项为一步的方向和目标点。",
-          "!type": "fn(destX: number, destY: number, stepPostfix: [{x: number, y: number, direction: string}])"
+        setAutomaticRoute: {
+          "!doc":
+            '半自动寻路,用于鼠标或手指拖动<br/>例如:core.setAutomaticRoute(0, 0, [{direction: "right", x: 4, y: 9}, {direction: "right", x: 5, y: 9}]);<br/>destX: 鼠标或手指的起拖点横坐标<br/>destY: 鼠标或手指的起拖点纵坐标<br/>stepPostfix: 拖动轨迹的数组表示,每项为一步的方向和目标点。',
+          "!type":
+            "fn(destX: number, destY: number, stepPostfix: [{x: number, y: number, direction: string}])",
         },
-        "setHeroOpacity": {
+        setHeroOpacity: {
           "!doc": "改变勇士的不透明度",
-          "!type": "fn(opacity?: number, moveMode?: string, time?: number, callback?: fn())"
+          "!type":
+            "fn(opacity?: number, moveMode?: string, time?: number, callback?: fn())",
         },
-        "gatherFollowers": {
+        gatherFollowers: {
           "!doc": "立刻聚集所有的跟随者",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "getStatus": {
-          "!doc": "读取主角的某个属性,不包括百分比修正<br/>例如:core.getStatus('atk'); // 读取主角的攻击力<br/>name: 属性的英文名,其中'x'、'y'和'direction'会被特殊处理为 core.getHeroLoc(name),其他的会直接读取 core.status.hero[name]",
-          "!type": "fn(name: string) -> number"
+        getStatus: {
+          "!doc":
+            "读取主角的某个属性,不包括百分比修正<br/>例如:core.getStatus('atk'); // 读取主角的攻击力<br/>name: 属性的英文名,其中'x'、'y'和'direction'会被特殊处理为 core.getHeroLoc(name),其他的会直接读取 core.status.hero[name]",
+          "!type": "fn(name: string) -> number",
         },
-        "setHeroLoc": {
-          "!doc": "设置勇士位置<br/>值得注意的是,这句话虽然会使勇士改变位置,但并不会使界面重新绘制;<br/>如需立刻重新绘制地图还需调用:core.clearMap('hero'); core.drawHero(); 来对界面进行更新。<br/>例如:core.setHeroLoc('x', 5) // 将勇士当前位置的横坐标设置为5。<br/>name: 要设置的坐标属性<br/>value: 新值<br/>noGather: 是否聚集跟随者",
-          "!type": "fn(name: string, value: string|number, noGather?: bool)"
+        setHeroLoc: {
+          "!doc":
+            "设置勇士位置<br/>值得注意的是,这句话虽然会使勇士改变位置,但并不会使界面重新绘制;<br/>如需立刻重新绘制地图还需调用:core.clearMap('hero'); core.drawHero(); 来对界面进行更新。<br/>例如:core.setHeroLoc('x', 5) // 将勇士当前位置的横坐标设置为5。<br/>name: 要设置的坐标属性<br/>value: 新值<br/>noGather: 是否聚集跟随者",
+          "!type": "fn(name: string, value: string|number, noGather?: bool)",
         },
-        "getLvName": {
-          "!doc": "根据级别的数字获取对应的名称,后者定义在全塔属性<br/>例如:core.getLvName(); // 获取主角当前级别的名称,如“下级佣兵”<br/>lv: 级别的数字,不填则视为主角当前的级别<br/>返回值:级别的名称,如果不存在就还是返回数字",
-          "!type": "fn(lv?: number) -> string|number"
+        getLvName: {
+          "!doc":
+            "根据级别的数字获取对应的名称,后者定义在全塔属性<br/>例如:core.getLvName(); // 获取主角当前级别的名称,如“下级佣兵”<br/>lv: 级别的数字,不填则视为主角当前的级别<br/>返回值:级别的名称,如果不存在就还是返回数字",
+          "!type": "fn(lv?: number) -> string|number",
         },
-        "getNextLvUpNeed": {
-          "!doc": "获得下次升级需要的经验值。<br/>升级扣除模式下会返回经验差值;非扣除模式下会返回总共需要的经验值。<br/>如果无法进行下次升级,返回null。",
-          "!type": "fn() -> number"
+        getNextLvUpNeed: {
+          "!doc":
+            "获得下次升级需要的经验值。<br/>升级扣除模式下会返回经验差值;非扣除模式下会返回总共需要的经验值。<br/>如果无法进行下次升级,返回null。",
+          "!type": "fn() -> number",
         },
-        "addStatus": {
-          "!doc": "增减主角的某个属性,等价于core.setStatus(name, core.getStatus(name) + value)<br/>例如:core.addStatus('atk', 100'); // 给主角攻击力加100<br/>name: 属性的英文名<br/>value: 属性的增量",
-          "!type": "fn(name: string, value: number)"
+        addStatus: {
+          "!doc":
+            "增减主角的某个属性,等价于core.setStatus(name, core.getStatus(name) + value)<br/>例如:core.addStatus('atk', 100'); // 给主角攻击力加100<br/>name: 属性的英文名<br/>value: 属性的增量",
+          "!type": "fn(name: string, value: number)",
         },
-        "speedUpReplay": {
+        speedUpReplay: {
           "!doc": "加速播放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "loadData": {
+        loadData: {
           "!doc": "从本地读档",
-          "!type": "fn(data?: ?, callback?: fn())"
+          "!type": "fn(data?: ?, callback?: fn())",
         },
-        "debug": {
-          "!doc": "开启调试模式, 此模式下可以按Ctrl键进行穿墙, 并忽略一切事件。<br/>此模式下不可回放录像和上传成绩。",
-          "!type": "fn()"
+        debug: {
+          "!doc":
+            "开启调试模式, 此模式下可以按Ctrl键进行穿墙, 并忽略一切事件。<br/>此模式下不可回放录像和上传成绩。",
+          "!type": "fn()",
         },
-        "moveOneStep": {
-          "!doc": "每移动一格后执行的事件<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(callback?: fn())"
+        moveOneStep: {
+          "!doc":
+            "每移动一格后执行的事件<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type": "fn(callback?: fn())",
         },
-        "clearStatus": {
+        clearStatus: {
           "!doc": "清除游戏状态和数据",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "updateFollowers": {
+        updateFollowers: {
           "!doc": "更新跟随者坐标",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "waitHeroToStop": {
-          "!doc": "等待主角停下<br/>例如:core.waitHeroToStop(core.vibrate); // 等待主角停下,然后视野左右抖动1秒<br/>callback: 主角停止后的回调函数",
-          "!type": "fn(callback?: fn())"
+        waitHeroToStop: {
+          "!doc":
+            "等待主角停下<br/>例如:core.waitHeroToStop(core.vibrate); // 等待主角停下,然后视野左右抖动1秒<br/>callback: 主角停止后的回调函数",
+          "!type": "fn(callback?: fn())",
         },
-        "hideStatusBar": {
+        hideStatusBar: {
           "!doc": "隐藏状态栏<br/>showToolbox: 是否不隐藏竖屏工具栏",
-          "!type": "fn(showToolbox?: bool)"
+          "!type": "fn(showToolbox?: bool)",
         },
-        "getBuff": {
-          "!doc": "读取主角某个属性的百分比修正倍率,初始值为1<br/>例如:core.getBuff('atk'); // 主角当前能发挥出多大比例的攻击力<br/>name: 属性的英文名",
-          "!type": "fn(name: string) -> number"
+        getBuff: {
+          "!doc":
+            "读取主角某个属性的百分比修正倍率,初始值为1<br/>例如:core.getBuff('atk'); // 主角当前能发挥出多大比例的攻击力<br/>name: 属性的英文名",
+          "!type": "fn(name: string) -> number",
         },
-        "triggerDebuff": {
-          "!doc": "获得或移除毒衰咒效果<br/>action: 要获得还是移除,'get'为获得,'remove'为移除<br/>type: 获得或移除的内容(poison/weak/curse),可以为字符串或数组",
-          "!type": "fn(action: string, type: string|[string])"
+        triggerDebuff: {
+          "!doc":
+            "获得或移除毒衰咒效果<br/>action: 要获得还是移除,'get'为获得,'remove'为移除<br/>type: 获得或移除的内容(poison/weak/curse),可以为字符串或数组",
+          "!type": "fn(action: string, type: string|[string])",
         },
-        "setToolbarButton": {
+        setToolbarButton: {
           "!doc": "改变工具栏为按钮1-8",
-          "!type": "fn(useButton?: bool)"
+          "!type": "fn(useButton?: bool)",
         },
-        "getSaves": {
+        getSaves: {
           "!doc": "获得某些存档内容",
-          "!type": "fn(ids?: ?, callback?: fn())"
+          "!type": "fn(ids?: ?, callback?: fn())",
         },
-        "replay": {
+        replay: {
           "!doc": "回放下一个操作",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "getStatusOrDefault": {
+        getStatusOrDefault: {
           "!doc": "从status中获得属性,如果不存在则从勇士属性中获取",
-          "!type": "fn(status?: ?, name?: string)"
+          "!type": "fn(status?: ?, name?: string)",
         },
-        "unregisterReplayAction": {
+        unregisterReplayAction: {
           "!doc": "注销一个录像行为",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "unregisterWeather": {
+        unregisterWeather: {
           "!doc": "注销一个天气",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "setBuff": {
-          "!doc": "设置主角某个属性的百分比修正倍率,初始值为1,<br/>倍率存放在flag: '__'+name+'_buff__' 中<br/>例如:core.setBuff('atk', 0.5); // 主角能发挥出的攻击力减半<br/>name: 属性的英文名,请注意只能用于数值类属性哦,否则随后的乘法会得到NaN<br/>value: 新的百分比修正倍率,不填(效果上)视为1",
-          "!type": "fn(name: string, value: number)"
+        setBuff: {
+          "!doc":
+            "设置主角某个属性的百分比修正倍率,初始值为1,<br/>倍率存放在flag: '__'+name+'_buff__' 中<br/>例如:core.setBuff('atk', 0.5); // 主角能发挥出的攻击力减半<br/>name: 属性的英文名,请注意只能用于数值类属性哦,否则随后的乘法会得到NaN<br/>value: 新的百分比修正倍率,不填(效果上)视为1",
+          "!type": "fn(name: string, value: number)",
         },
-        "continueAutomaticRoute": {
+        continueAutomaticRoute: {
           "!doc": "继续剩下的自动寻路操作",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setAutoHeroMove": {
-          "!doc": "连续行走<br/>例如:core.setAutoHeroMove([{direction: \"up\", step: 1}, {direction: \"left\", step: 3}]); // 上左左左<br/>steps: 压缩的步伐数组,每项表示朝某方向走多少步",
-          "!type": "fn(steps: [?])"
+        setAutoHeroMove: {
+          "!doc":
+            '连续行走<br/>例如:core.setAutoHeroMove([{direction: "up", step: 1}, {direction: "left", step: 3}]); // 上左左左<br/>steps: 压缩的步伐数组,每项表示朝某方向走多少步',
+          "!type": "fn(steps: [?])",
         },
-        "unregisterResize": {
+        unregisterResize: {
           "!doc": "注销一个resize函数",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "saveAndStopAutomaticRoute": {
+        saveAndStopAutomaticRoute: {
           "!doc": "保存剩下的寻路,并停止",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "hideStartAnimate": {
-          "!doc": "淡出标题画面<br/>例如:core.hideStartAnimate(core.startGame); // 淡出标题画面并开始新游戏,跳过难度选择<br/>callback: 标题画面完全淡出后的回调函数",
-          "!type": "fn(callback?: fn())"
+        hideStartAnimate: {
+          "!doc":
+            "淡出标题画面<br/>例如:core.hideStartAnimate(core.startGame); // 淡出标题画面并开始新游戏,跳过难度选择<br/>callback: 标题画面完全淡出后的回调函数",
+          "!type": "fn(callback?: fn())",
         },
-        "getAllSaves": {
+        getAllSaves: {
           "!doc": "获得所有存档内容",
-          "!type": "fn(callback?: fn())"
+          "!type": "fn(callback?: fn())",
         },
-        "updateHeroIcon": {
+        updateHeroIcon: {
           "!doc": "更新状态栏的勇士图标",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "setMusicBtn": {
+        setMusicBtn: {
           "!doc": "设置音乐图标的显隐状态",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "isPlaying": {
+        isPlaying: {
           "!doc": "游戏是否已经开始",
-          "!type": "fn() -> bool"
+          "!type": "fn() -> bool",
         },
-        "triggerBgm": {
+        triggerBgm: {
           "!doc": "开启或关闭背景音乐的播放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "moveHero": {
-          "!doc": "连续前进,不撞南墙不回头<br/>例如:core.moveHero(); // 连续前进<br/>direction: 可选,如果设置了就会先转身到该方向<br/>callback: 可选,如果设置了就只走一步<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(direction?: string, callback?: fn())"
+        moveHero: {
+          "!doc":
+            "连续前进,不撞南墙不回头<br/>例如:core.moveHero(); // 连续前进<br/>direction: 可选,如果设置了就会先转身到该方向<br/>callback: 可选,如果设置了就只走一步<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type": "fn(direction?: string, callback?: fn())",
         },
-        "getRealStatusOrDefault": {
-          "!doc": "从status中获得实际属性(增幅后的),如果不存在则从勇士属性中获取",
-          "!type": "fn(status?: ?, name?: string)"
+        getRealStatusOrDefault: {
+          "!doc":
+            "从status中获得实际属性(增幅后的),如果不存在则从勇士属性中获取",
+          "!type": "fn(status?: ?, name?: string)",
         },
-        "getStatusLabel": {
+        getStatusLabel: {
           "!doc": "获得某个状态的名字,如atk->攻击,def->防御等",
-          "!type": "fn(name: string) -> string"
+          "!type": "fn(name: string) -> string",
         },
-        "removeSave": {
+        removeSave: {
           "!doc": "删除某个存档",
-          "!type": "fn(index?: number, callback?: fn())"
+          "!type": "fn(index?: number, callback?: fn())",
         },
-        "registerAnimationFrame": {
-          "!doc": "注册一个 animationFrame<br/>name: 名称,可用来作为注销使用<br/>needPlaying: 是否只在游戏运行时才执行(在标题界面不执行)<br/>func: 要执行的函数,或插件中的函数名;可接受timestamp(从页面加载完毕到当前所经过的时间)作为参数",
-          "!type": "fn(name: string, needPlaying: bool, func?: fn(timestamp: number))"
+        registerAnimationFrame: {
+          "!doc":
+            "注册一个 animationFrame<br/>name: 名称,可用来作为注销使用<br/>needPlaying: 是否只在游戏运行时才执行(在标题界面不执行)<br/>func: 要执行的函数,或插件中的函数名;可接受timestamp(从页面加载完毕到当前所经过的时间)作为参数",
+          "!type":
+            "fn(name: string, needPlaying: bool, func?: fn(timestamp: number))",
         },
-        "getHeroLoc": {
-          "!doc": "读取主角的位置和/或朝向<br/>例如:core.getHeroLoc(); // 读取主角的位置和朝向<br/>name: 要读取横坐标还是纵坐标还是朝向还是都读取<br/>返回值:name ? core.status.hero.loc[name] : core.status.hero.loc",
-          "!type": "fn(name: string) -> string|number"
+        getHeroLoc: {
+          "!doc":
+            "读取主角的位置和/或朝向<br/>例如:core.getHeroLoc(); // 读取主角的位置和朝向<br/>name: 要读取横坐标还是纵坐标还是朝向还是都读取<br/>返回值:name ? core.status.hero.loc[name] : core.status.hero.loc",
+          "!type": "fn(name: string) -> string|number",
         },
-        "stopAutomaticRoute": {
+        stopAutomaticRoute: {
           "!doc": "停止自动寻路操作",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setWeather": {
-          "!doc": "设置天气,不计入存档。如需长期生效请使用core.events._action_setWeather()函数<br/>例如:core.setWeather('fog', 10); // 设置十级大雾天<br/>type: 新天气的类型,不填视为晴天<br/>level: 新天气(晴天除外)的级别,必须为不大于10的正整数,不填视为5",
-          "!type": "fn(type?: string, level?: number)"
+        setWeather: {
+          "!doc":
+            "设置天气,不计入存档。如需长期生效请使用core.events._action_setWeather()函数<br/>例如:core.setWeather('fog', 10); // 设置十级大雾天<br/>type: 新天气的类型,不填视为晴天<br/>level: 新天气(晴天除外)的级别,必须为不大于10的正整数,不填视为5",
+          "!type": "fn(type?: string, level?: number)",
         },
-        "updateStatusBar": {
-          "!doc": "刷新状态栏和地图显伤<br/>doNotCheckAutoEvents: 是否不检查自动事件",
-          "!type": "fn(doNotCheckAutoEvents?: bool, immediate?: bool)"
+        updateStatusBar: {
+          "!doc":
+            "刷新状态栏和地图显伤<br/>doNotCheckAutoEvents: 是否不检查自动事件",
+          "!type": "fn(doNotCheckAutoEvents?: bool, immediate?: bool)",
         },
-        "autosave": {
+        autosave: {
           "!doc": "自动存档",
-          "!type": "fn(removeLast?: bool)"
+          "!type": "fn(removeLast?: bool)",
         },
-        "clearStatusBar": {
+        clearStatusBar: {
           "!doc": "清空状态栏",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "moveAction": {
-          "!doc": "尝试前进一步,如果面前不可被踏入就会直接触发该点事件<br/>请勿直接使用此函数,如有需要请使用「勇士前进一步或撞击」事件<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(callback?: fn())"
+        moveAction: {
+          "!doc":
+            "尝试前进一步,如果面前不可被踏入就会直接触发该点事件<br/>请勿直接使用此函数,如有需要请使用「勇士前进一步或撞击」事件<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type": "fn(callback?: fn())",
         },
-        "hasFlag": {
-          "!doc": "判定一个flag变量是否存在且不为false、0、''、null、undefined和NaN<br/>例如:core.hasFlag('poison'); // 判断主角当前是否中毒<br/>name: 变量名,支持中文<br/>此函数等价于 !!core.getFlag(name)",
-          "!type": "fn(name: string) -> bool"
+        hasFlag: {
+          "!doc":
+            "判定一个flag变量是否存在且不为false、0、''、null、undefined和NaN<br/>例如:core.hasFlag('poison'); // 判断主角当前是否中毒<br/>name: 变量名,支持中文<br/>此函数等价于 !!core.getFlag(name)",
+          "!type": "fn(name: string) -> bool",
         },
-        "rewindReplay": {
+        rewindReplay: {
           "!doc": "回退到上一个录像节点",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "playBgm": {
-          "!doc": "播放背景音乐,中途开播但不计入存档且只会持续到下次场景切换。如需长期生效请将背景音乐的文件名赋值给flags.__bgm__<br/>例如:core.playBgm('bgm.mp3', 30); // 播放bgm.mp3,并跳过前半分钟<br/>bgm: 背景音乐的文件名,支持全塔属性中映射前的中文名<br/>startTime: 跳过前多少秒,不填则不跳过",
-          "!type": "fn(bgm: string, startTime?: number)"
+        playBgm: {
+          "!doc":
+            "播放背景音乐,中途开播但不计入存档且只会持续到下次场景切换。如需长期生效请将背景音乐的文件名赋值给flags.__bgm__<br/>例如:core.playBgm('bgm.mp3', 30); // 播放bgm.mp3,并跳过前半分钟<br/>bgm: 背景音乐的文件名,支持全塔属性中映射前的中文名<br/>startTime: 跳过前多少秒,不填则不跳过",
+          "!type": "fn(bgm: string, startTime?: number)",
         },
-        "isReplaying": {
+        isReplaying: {
           "!doc": "是否正在播放录像",
-          "!type": "fn() -> bool"
+          "!type": "fn() -> bool",
         },
-        "isMoving": {
+        isMoving: {
           "!doc": "当前是否正在移动",
-          "!type": "fn() -> bool"
+          "!type": "fn() -> bool",
         },
-        "getSaveIndexes": {
+        getSaveIndexes: {
           "!doc": "获得所有存在存档的存档位",
-          "!type": "fn(callback?: fn())"
+          "!type": "fn(callback?: fn())",
         },
-        "unlockControl": {
+        unlockControl: {
           "!doc": "解锁用户控制行为",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "syncSave": {
+        syncSave: {
           "!doc": "同步存档到服务器",
-          "!type": "fn(type?: string)"
+          "!type": "fn(type?: string)",
         },
-        "removeFlag": {
+        removeFlag: {
           "!doc": "删除某个flag/变量",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "registerResize": {
-          "!doc": "注册一个resize函数<br/>name: 名称,可供注销使用<br/>func: 可以是一个函数,或者是插件中的函数名;可以接受obj参数,详见resize函数。",
-          "!type": "fn(name: string, func: fn(obj: ?))"
+        registerResize: {
+          "!doc":
+            "注册一个resize函数<br/>name: 名称,可供注销使用<br/>func: 可以是一个函数,或者是插件中的函数名;可以接受obj参数,详见resize函数。",
+          "!type": "fn(name: string, func: fn(obj: ?))",
         },
-        "registerWeather": {
-          "!doc": "注册一个天气<br/>name: 要注册的天气名<br/>initFunc: 当切换到此天气时的初始化;接受level(天气等级)为参数;可用于创建多个节点(如初始化雪花)<br/>frameFunc: 每帧的天气效果变化;可接受timestamp(从页面加载完毕到当前所经过的时间)和level(天气等级)作为参数<br/>天气应当仅在weather层进行绘制,推荐使用core.animateFrame.weather.nodes用于节点信息。",
-          "!type": "fn(name: string, initFunc: fn(level: number), frameFunc?: fn(timestamp: number, level: number))"
+        registerWeather: {
+          "!doc":
+            "注册一个天气<br/>name: 要注册的天气名<br/>initFunc: 当切换到此天气时的初始化;接受level(天气等级)为参数;可用于创建多个节点(如初始化雪花)<br/>frameFunc: 每帧的天气效果变化;可接受timestamp(从页面加载完毕到当前所经过的时间)和level(天气等级)作为参数<br/>天气应当仅在weather层进行绘制,推荐使用core.animateFrame.weather.nodes用于节点信息。",
+          "!type":
+            "fn(name: string, initFunc: fn(level: number), frameFunc?: fn(timestamp: number, level: number))",
         },
-        "stopReplay": {
+        stopReplay: {
           "!doc": "停止播放",
-          "!type": "fn(force?: bool)"
+          "!type": "fn(force?: bool)",
         },
-        "turnHero": {
-          "!doc": "主角转向并计入录像,不会导致跟随者聚集,会导致视野重置到以主角为中心<br/>例如:core.turnHero(); // 主角顺时针旋转90°,即单击主角或按下Z键的效果<br/>direction: 主角的新朝向,可为 up, down, left, right, :left, :right, :back 七种之一",
-          "!type": "fn(direction?: string)"
+        turnHero: {
+          "!doc":
+            "主角转向并计入录像,不会导致跟随者聚集,会导致视野重置到以主角为中心<br/>例如:core.turnHero(); // 主角顺时针旋转90°,即单击主角或按下Z键的效果<br/>direction: 主角的新朝向,可为 up, down, left, right, :left, :right, :back 七种之一",
+          "!type": "fn(direction?: string)",
         },
-        "resumeReplay": {
+        resumeReplay: {
           "!doc": "恢复播放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "resize": {
+        resize: {
           "!doc": "屏幕分辨率改变后重新自适应",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "getSave": {
+        getSave: {
           "!doc": "获得某个存档内容",
-          "!type": "fn(index?: number, callback?: fn(data: ?))"
+          "!type": "fn(index?: number, callback?: fn(data: ?))",
         },
-        "setViewport": {
-          "!doc": "设置视野范围<br/>px,py: 左上角相对大地图的像素坐标,不需要为32倍数",
-          "!type": "fn(px?: number, py?: number)"
+        setViewport: {
+          "!doc":
+            "设置视野范围<br/>px,py: 左上角相对大地图的像素坐标,不需要为32倍数",
+          "!type": "fn(px?: number, py?: number)",
         },
-        "chooseReplayFile": {
+        chooseReplayFile: {
           "!doc": "选择录像文件",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "lockControl": {
+        lockControl: {
           "!doc": "锁定用户控制,常常用于事件处理",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "updateCheckBlock": {
+        updateCheckBlock: {
           "!doc": "更新领域、夹击、阻击的伤害地图",
-          "!type": "fn(floorId?: string)"
+          "!type": "fn(floorId?: string)",
         },
-        "checkBlock": {
+        checkBlock: {
           "!doc": "检查并执行领域、夹击、阻击事件",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "clearAutomaticRouteNode": {
+        clearAutomaticRouteNode: {
           "!doc": "清除自动寻路路线",
-          "!type": "fn(x?: number, y?: number)"
+          "!type": "fn(x?: number, y?: number)",
         },
-        "getFlag": {
-          "!doc": "读取一个flag变量<br/>name: 变量名,支持中文<br/>defaultValue: 当变量不存在时的返回值,可选(事件流中默认填0)。",
-          "!type": "fn(name: string, defaultValue?: ?)"
+        getFlag: {
+          "!doc":
+            "读取一个flag变量<br/>name: 变量名,支持中文<br/>defaultValue: 当变量不存在时的返回值,可选(事件流中默认填0)。",
+          "!type": "fn(name: string, defaultValue?: ?)",
         },
-        "getNakedStatus": {
+        getNakedStatus: {
           "!doc": "获得勇士原始属性(无装备和衰弱影响)",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "nearHero": {
-          "!doc": "判定主角是否身处某个点的锯齿领域(取曼哈顿距离)<br/>例如:core.nearHero(6, 6, 6); // 判定主角是否身处点(6,6)的半径为6的锯齿领域<br/>x: 领域的中心横坐标<br/>y: 领域的中心纵坐标<br/>n: 领域的半径,不填视为1",
-          "!type": "fn(x: number, y: number, n?: number) -> bool"
+        nearHero: {
+          "!doc":
+            "判定主角是否身处某个点的锯齿领域(取曼哈顿距离)<br/>例如:core.nearHero(6, 6, 6); // 判定主角是否身处点(6,6)的半径为6的锯齿领域<br/>x: 领域的中心横坐标<br/>y: 领域的中心纵坐标<br/>n: 领域的半径,不填视为1",
+          "!type": "fn(x: number, y: number, n?: number) -> bool",
         },
-        "stepReplay": {
+        stepReplay: {
           "!doc": "单步播放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "hasSave": {
+        hasSave: {
           "!doc": "判断某个存档位是否存在存档",
-          "!type": "fn(index?: number) -> bool"
+          "!type": "fn(index?: number) -> bool",
         },
-        "showStartAnimate": {
-          "!doc": "进入标题画面<br/>例如:core.showStartAnimate(); // 重启游戏但不重置bgm<br/>noAnimate: 可选,true表示不由黑屏淡入而是立即亮屏<br/>callback: 可选,完全亮屏后的回调函数",
-          "!type": "fn(noAnimate?: bool, callback?: fn())"
+        showStartAnimate: {
+          "!doc":
+            "进入标题画面<br/>例如:core.showStartAnimate(); // 重启游戏但不重置bgm<br/>noAnimate: 可选,true表示不由黑屏淡入而是立即亮屏<br/>callback: 可选,完全亮屏后的回调函数",
+          "!type": "fn(noAnimate?: bool, callback?: fn())",
         },
-        "moveViewport": {
+        moveViewport: {
           "!doc": "移动视野范围",
-          "!type": "fn(x: number, y: number, moveMode?: string, time?: number, callback?: fn())"
+          "!type":
+            "fn(x: number, y: number, moveMode?: string, time?: number, callback?: fn())",
         },
-        "syncLoad": {
+        syncLoad: {
           "!doc": "从服务器加载存档",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setHeroMoveInterval": {
+        setHeroMoveInterval: {
           "!doc": "设置行走的效果动画",
-          "!type": "fn(callback?: fn())"
+          "!type": "fn(callback?: fn())",
         },
-        "registerReplayAction": {
-          "!doc": "注册一个录像行为<br/>name: 自定义名称,可用于注销使用<br/>func: 具体执行录像的函数,可为一个函数或插件中的函数名;<br/>需要接受一个action参数,代表录像回放时的下一个操作<br/>func返回true代表成功处理了此录像行为,false代表没有处理此录像行为。",
-          "!type": "fn(name: string, func: fn(action?: string) -> bool)"
+        registerReplayAction: {
+          "!doc":
+            "注册一个录像行为<br/>name: 自定义名称,可用于注销使用<br/>func: 具体执行录像的函数,可为一个函数或插件中的函数名;<br/>需要接受一个action参数,代表录像回放时的下一个操作<br/>func返回true代表成功处理了此录像行为,false代表没有处理此录像行为。",
+          "!type": "fn(name: string, func: fn(action?: string) -> bool)",
         },
-        "checkAutosave": {
+        checkAutosave: {
           "!doc": "实际将自动存档写入存储",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "resumeBgm": {
+        resumeBgm: {
           "!doc": "恢复背景音乐的播放<br/>resumeTime: 从哪一秒开始恢复播放",
-          "!type": "fn(resumeTime?: number)"
+          "!type": "fn(resumeTime?: number)",
         },
-        "setGameCanvasTranslate": {
+        setGameCanvasTranslate: {
           "!doc": "设置大地图的偏移量",
-          "!type": "fn(ctx: string|CanvasRenderingContext2D, x: number, y: number)"
+          "!type":
+            "fn(ctx: string|CanvasRenderingContext2D, x: number, y: number)",
         },
-        "checkBgm": {
+        checkBgm: {
           "!doc": "检查bgm状态",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setDisplayScale": {
+        setDisplayScale: {
           "!doc": "设置屏幕放缩",
-          "!type": "fn(delta: number)"
+          "!type": "fn(delta: number)",
         },
-        "speedDownReplay": {
+        speedDownReplay: {
           "!doc": "减速播放",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "getRealStatus": {
-          "!doc": "计算主角的某个属性,包括百分比修正<br/>例如:core.getRealStatus('atk'); // 计算主角的攻击力,包括百分比修正。战斗使用的就是这个值<br/>name: 属性的英文名,请注意只能用于数值类属性哦,否则乘法会得到NaN",
-          "!type": "fn(name: string)"
+        getRealStatus: {
+          "!doc":
+            "计算主角的某个属性,包括百分比修正<br/>例如:core.getRealStatus('atk'); // 计算主角的攻击力,包括百分比修正。战斗使用的就是这个值<br/>name: 属性的英文名,请注意只能用于数值类属性哦,否则乘法会得到NaN",
+          "!type": "fn(name: string)",
         },
-        "saveData": {
+        saveData: {
           "!doc": "存档到本地",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "unregisterAnimationFrame": {
+        unregisterAnimationFrame: {
           "!doc": "注销一个animationFrame",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "tryMoveDirectly": {
-          "!doc": "尝试瞬移,如果该点有图块/事件/阻激夹域捕则会瞬移到它旁边再走一步(不可踏入的话当然还是触发该点事件),这一步的方向优先和瞬移前主角的朝向一致<br/>例如:core.tryMoveDirectly(6, 0); // 尝试瞬移到地图顶部的正中央,以样板0层为例,实际效果是瞬移到了上楼梯下面一格然后向上走一步并触发上楼事件<br/>destX: 目标点的横坐标<br/>destY: 目标点的纵坐标",
-          "!type": "fn(destX: number, destY: number)"
+        tryMoveDirectly: {
+          "!doc":
+            "尝试瞬移,如果该点有图块/事件/阻激夹域捕则会瞬移到它旁边再走一步(不可踏入的话当然还是触发该点事件),这一步的方向优先和瞬移前主角的朝向一致<br/>例如:core.tryMoveDirectly(6, 0); // 尝试瞬移到地图顶部的正中央,以样板0层为例,实际效果是瞬移到了上楼梯下面一格然后向上走一步并触发上楼事件<br/>destX: 目标点的横坐标<br/>destY: 目标点的纵坐标",
+          "!type": "fn(destX: number, destY: number)",
         },
-        "moveDirectly": {
+        moveDirectly: {
           "!doc": "瞬间移动",
-          "!type": "fn(destX?: number, destY?: number, ignoreSteps?: number)"
+          "!type": "fn(destX?: number, destY?: number, ignoreSteps?: number)",
         },
-        "clearRouteFolding": {
+        clearRouteFolding: {
           "!doc": "清空录像折叠信息",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "checkRouteFolding": {
+        checkRouteFolding: {
           "!doc": "检查录像折叠信息",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setSwitch": {
+        setSwitch: {
           "!doc": "设置某个独立开关",
-          "!type": "fn(x: number, y: number, floorId: string, name: string, value: ?)"
+          "!type":
+            "fn(x: number, y: number, floorId: string, name: string, value: ?)",
         },
-        "getSwitch": {
+        getSwitch: {
           "!doc": "获得某个独立开关",
-          "!type": "fn(x: number, y: number, floorId: string, name: string, defaultValue?: ?)"
+          "!type":
+            "fn(x: number, y: number, floorId: string, name: string, defaultValue?: ?)",
         },
-        "addSwitch": {
+        addSwitch: {
           "!doc": "增加某个独立开关",
-          "!type": "fn(x: number, y: number, floorId: string, name: string, value: number)"
+          "!type":
+            "fn(x: number, y: number, floorId: string, name: string, value: number)",
         },
-        "removeSwitch": {
+        removeSwitch: {
           "!doc": "删除某个独立开关",
-          "!type": "fn(x: number, y: number, floorId: string, name: string)"
+          "!type": "fn(x: number, y: number, floorId: string, name: string)",
         },
-        "removeSwitch": {
+        removeSwitch: {
           "!doc": "判定某个独立开关",
-          "!type": "fn(x: number, y: number, floorId: string, name: string) -> bool"
-        }
+          "!type":
+            "fn(x: number, y: number, floorId: string, name: string) -> bool",
+        },
       },
-      "icons": {
+      icons: {
         "!doc": "图标信息",
-        "getTilesetOffset": {
+        getTilesetOffset: {
           "!doc": "根据图块数字或ID获得所在的tileset和坐标信息",
-          "!type": "fn(id?: string) -> {image: ?, x: number, y: number}"
+          "!type": "fn(id?: string) -> {image: ?, x: number, y: number}",
         },
-        "getClsFromId": {
+        getClsFromId: {
           "!doc": "根据ID获得其图标类型",
-          "!type": "fn(id?: string) -> string"
+          "!type": "fn(id?: string) -> string",
         },
-        "getAllIconIds": {
+        getAllIconIds: {
           "!doc": "获得所有图标的ID",
-          "!type": "fn() -> [string]"
+          "!type": "fn() -> [string]",
         },
-        "getIcons": {
+        getIcons: {
           "!doc": "获得所有图标类型",
-          "!type": "fn()"
-        }
+          "!type": "fn()",
+        },
       },
-      "items": {
+      items: {
         "!doc": "道具相关的函数",
-        "getEquip": {
-          "!doc": "检查主角某种类型的装备目前是什么<br/>例如:core.getEquip(1) // 主角目前装备了什么盾牌<br/>equipType: 装备类型,自然数<br/>返回值:装备id,null表示未穿戴",
-          "!type": "fn(equipType: number) -> string"
+        getEquip: {
+          "!doc":
+            "检查主角某种类型的装备目前是什么<br/>例如:core.getEquip(1) // 主角目前装备了什么盾牌<br/>equipType: 装备类型,自然数<br/>返回值:装备id,null表示未穿戴",
+          "!type": "fn(equipType: number) -> string",
         },
-        "loadEquip": {
-          "!doc": "尝试穿上某件背包里面的装备并提示<br/>例如:core.loadEquip('sword5') // 尝试装备上背包里面的神圣剑,无回调<br/>equipId: 装备id<br/>callback: 穿戴成功或失败后的回调函数",
-          "!type": "fn(equipId: string, callback?: fn())"
+        loadEquip: {
+          "!doc":
+            "尝试穿上某件背包里面的装备并提示<br/>例如:core.loadEquip('sword5') // 尝试装备上背包里面的神圣剑,无回调<br/>equipId: 装备id<br/>callback: 穿戴成功或失败后的回调函数",
+          "!type": "fn(equipId: string, callback?: fn())",
         },
-        "itemCount": {
-          "!doc": "统计某种道具的持有量<br/>例如:core.itemCount('yellowKey') // 持有多少把黄钥匙<br/>itemId: 道具id<br/>返回值:该种道具的持有量,不包括已穿戴的装备",
-          "!type": "fn(itemId: string) -> number"
+        itemCount: {
+          "!doc":
+            "统计某种道具的持有量<br/>例如:core.itemCount('yellowKey') // 持有多少把黄钥匙<br/>itemId: 道具id<br/>返回值:该种道具的持有量,不包括已穿戴的装备",
+          "!type": "fn(itemId: string) -> number",
         },
-        "getItems": {
+        getItems: {
           "!doc": "获得所有道具",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "canUseItem": {
-          "!doc": "检查能否使用某种道具<br/>例如:core.canUseItem('pickaxe') // 能否使用破墙镐<br/>itemId: 道具id<br/>返回值:true表示可以使用",
-          "!type": "fn(itemId: string) -> bool"
+        canUseItem: {
+          "!doc":
+            "检查能否使用某种道具<br/>例如:core.canUseItem('pickaxe') // 能否使用破墙镐<br/>itemId: 道具id<br/>返回值:true表示可以使用",
+          "!type": "fn(itemId: string) -> bool",
         },
-        "hasItem": {
-          "!doc": "检查主角是否持有某种道具(不包括已穿戴的装备)<br/>例如:core.hasItem('yellowKey') // 主角是否持有黄钥匙<br/>itemId: 道具id<br/>返回值:true表示持有",
-          "!type": "fn(itemId: string) -> bool"
+        hasItem: {
+          "!doc":
+            "检查主角是否持有某种道具(不包括已穿戴的装备)<br/>例如:core.hasItem('yellowKey') // 主角是否持有黄钥匙<br/>itemId: 道具id<br/>返回值:true表示持有",
+          "!type": "fn(itemId: string) -> bool",
         },
-        "addItem": {
-          "!doc": "静默增减某种道具的持有量 不会更新游戏画面或是显示提示<br/>例如:core.addItem('yellowKey', -2) // 没收两把黄钥匙<br/>itemId: 道具id<br/>itemNum: 增加量,负数表示没收",
-          "!type": "fn(itemId: string, itemNum?: number)"
+        addItem: {
+          "!doc":
+            "静默增减某种道具的持有量 不会更新游戏画面或是显示提示<br/>例如:core.addItem('yellowKey', -2) // 没收两把黄钥匙<br/>itemId: 道具id<br/>itemNum: 增加量,负数表示没收",
+          "!type": "fn(itemId: string, itemNum?: number)",
         },
-        "unloadEquip": {
-          "!doc": "脱下某个类型的装备<br/>例如:core.unloadEquip(1) // 卸下盾牌,无回调<br/>equipType: 装备类型编号,自然数<br/>callback: 卸下装备后的回调函数",
-          "!type": "fn(equipType: number, callback?: fn())"
+        unloadEquip: {
+          "!doc":
+            "脱下某个类型的装备<br/>例如:core.unloadEquip(1) // 卸下盾牌,无回调<br/>equipType: 装备类型编号,自然数<br/>callback: 卸下装备后的回调函数",
+          "!type": "fn(equipType: number, callback?: fn())",
         },
-        "quickLoadEquip": {
-          "!doc": "快速换装<br/>例如:core.quickLoadEquip(1) // 快速换上1号套装<br/>index: 套装编号,自然数",
-          "!type": "fn(index: number)"
+        quickLoadEquip: {
+          "!doc":
+            "快速换装<br/>例如:core.quickLoadEquip(1) // 快速换上1号套装<br/>index: 套装编号,自然数",
+          "!type": "fn(index: number)",
         },
-        "getItemEffect": {
-          "!doc": "即捡即用类的道具获得时的效果<br/>例如:core.getItemEffect('redPotion', 10) // 执行获得10瓶红血的效果<br/>itemId: 道具id<br/>itemNum: 道具数量,可选,默认为1",
-          "!type": "fn(itemId: string, itemNum?: number)"
+        getItemEffect: {
+          "!doc":
+            "即捡即用类的道具获得时的效果<br/>例如:core.getItemEffect('redPotion', 10) // 执行获得10瓶红血的效果<br/>itemId: 道具id<br/>itemNum: 道具数量,可选,默认为1",
+          "!type": "fn(itemId: string, itemNum?: number)",
         },
-        "quickSaveEquip": {
-          "!doc": "保存当前套装<br/>例如:core.quickSaveEquip(1) // 将当前套装保存为1号套装<br/>index: 套装编号,自然数",
-          "!type": "fn(index: number)"
+        quickSaveEquip: {
+          "!doc":
+            "保存当前套装<br/>例如:core.quickSaveEquip(1) // 将当前套装保存为1号套装<br/>index: 套装编号,自然数",
+          "!type": "fn(index: number)",
         },
-        "setItem": {
-          "!doc": "设置某种道具的持有量<br/>例如:core.setItem('yellowKey', 3) // 设置黄钥匙为3把<br/>itemId: 道具id<br/>itemNum: 新的持有量,可选,自然数,默认为0",
-          "!type": "fn(itemId: string, itemNum?: number)"
+        setItem: {
+          "!doc":
+            "设置某种道具的持有量<br/>例如:core.setItem('yellowKey', 3) // 设置黄钥匙为3把<br/>itemId: 道具id<br/>itemNum: 新的持有量,可选,自然数,默认为0",
+          "!type": "fn(itemId: string, itemNum?: number)",
         },
-        "compareEquipment": {
-          "!doc": "比较两件(类型可不同)装备的优劣<br/>例如:core.compareEquipment('sword5', 'shield5') // 比较神圣剑和神圣盾的优劣<br/>compareEquipId: 装备甲的id<br/>beComparedEquipId: 装备乙的id<br/>返回值:两装备的各属性差,甲减乙,0省略",
-          "!type": "fn(compareEquipId: string, beComparedEquipId: string) -> {value: ?, percentage: ?}"
+        compareEquipment: {
+          "!doc":
+            "比较两件(类型可不同)装备的优劣<br/>例如:core.compareEquipment('sword5', 'shield5') // 比较神圣剑和神圣盾的优劣<br/>compareEquipId: 装备甲的id<br/>beComparedEquipId: 装备乙的id<br/>返回值:两装备的各属性差,甲减乙,0省略",
+          "!type":
+            "fn(compareEquipId: string, beComparedEquipId: string) -> {value: ?, percentage: ?}",
         },
-        "removeItem": {
+        removeItem: {
           "!doc": "删除某个物品",
-          "!type": "fn(itemId?: string, itemNum?: number)"
+          "!type": "fn(itemId?: string, itemNum?: number)",
         },
-        "getEquipTypeById": {
-          "!doc": "判定某件装备的类型<br/>例如:core.getEquipTypeById('shield5') // 1(盾牌)<br/>equipId: 装备id<br/>返回值:类型编号,自然数",
-          "!type": "fn(equipId: string) -> number"
+        getEquipTypeById: {
+          "!doc":
+            "判定某件装备的类型<br/>例如:core.getEquipTypeById('shield5') // 1(盾牌)<br/>equipId: 装备id<br/>返回值:类型编号,自然数",
+          "!type": "fn(equipId: string) -> number",
         },
-        "getEquipTypeByName": {
+        getEquipTypeByName: {
           "!doc": "根据类型获得一个可用的装备孔",
-          "!type": "fn(name?: string)"
+          "!type": "fn(name?: string)",
         },
-        "useItem": {
-          "!doc": "使用一个道具<br/>例如:core.useItem('pickaxe', true) // 使用破墙镐,不计入录像,无回调<br/>itemId: 道具id<br/>noRoute: 是否不计入录像,快捷键使用的请填true,否则可省略<br/>callback: 道具使用完毕或使用失败后的回调函数",
-          "!type": "fn(itemId: string, noRoute?: bool, callback?: fn())"
+        useItem: {
+          "!doc":
+            "使用一个道具<br/>例如:core.useItem('pickaxe', true) // 使用破墙镐,不计入录像,无回调<br/>itemId: 道具id<br/>noRoute: 是否不计入录像,快捷键使用的请填true,否则可省略<br/>callback: 道具使用完毕或使用失败后的回调函数",
+          "!type": "fn(itemId: string, noRoute?: bool, callback?: fn())",
         },
-        "hasEquip": {
-          "!doc": "检查主角是否穿戴着某件装备<br/>例如:core.hasEquip('sword5') // 主角是否装备了神圣剑<br/>itemId: 装备id<br/>返回值:true表示已装备",
-          "!type": "fn(itemId: string) -> bool"
+        hasEquip: {
+          "!doc":
+            "检查主角是否穿戴着某件装备<br/>例如:core.hasEquip('sword5') // 主角是否装备了神圣剑<br/>itemId: 装备id<br/>返回值:true表示已装备",
+          "!type": "fn(itemId: string) -> bool",
         },
-        "getItemEffectTip": {
-          "!doc": "即捡即用类的道具获得时的额外提示<br/>例如:core.getItemEffectTip(redPotion) // (获得 红血瓶)',生命+100'<br/>itemId: 道具id<br/>返回值:图块属性itemEffectTip的内容",
-          "!type": "fn(itemId: string) -> string"
+        getItemEffectTip: {
+          "!doc":
+            "即捡即用类的道具获得时的额外提示<br/>例如:core.getItemEffectTip(redPotion) // (获得 红血瓶)',生命+100'<br/>itemId: 道具id<br/>返回值:图块属性itemEffectTip的内容",
+          "!type": "fn(itemId: string) -> string",
         },
-        "canEquip": {
-          "!doc": "检查能否穿上某件装备<br/>例如:core.canEquip('sword5', true) // 主角可以装备神圣剑吗,如果不能会有提示<br/>equipId: 装备id<br/>hint: 无法穿上时是否提示(比如是因为未持有还是别的什么原因)<br/>返回值:true表示可以穿上,false表示无法穿上",
-          "!type": "fn(equipId: string, hint?: bool) -> bool"
+        canEquip: {
+          "!doc":
+            "检查能否穿上某件装备<br/>例如:core.canEquip('sword5', true) // 主角可以装备神圣剑吗,如果不能会有提示<br/>equipId: 装备id<br/>hint: 无法穿上时是否提示(比如是因为未持有还是别的什么原因)<br/>返回值:true表示可以穿上,false表示无法穿上",
+          "!type": "fn(equipId: string, hint?: bool) -> bool",
+        },
+        setEquip: {
+          "!doc":
+            "设置某个装备的属性并计入存档<br/>例如:core.setEquip('sword1', 'value', 'atk', 300, '+='); // 设置铁剑的攻击力数值再加300<br/>equipId: 装备id<br/>valueType: 增幅类型,只能是value(数值)或percentage(百分比)<br/>name: 要修改的属性名称,如atk<br/>value: 要修改到的属性数值<br/>operator: 操作符,可选,如+=表示在原始值上增加<br/>prefix: 独立开关前缀,一般不需要",
+          "!type":
+            "fn(equipId: string, valueType: string, name: string, value: ?, operator?: string, prefix?: string)",
         },
-        "setEquip": {
-          "!doc": "设置某个装备的属性并计入存档<br/>例如:core.setEquip('sword1', 'value', 'atk', 300, '+='); // 设置铁剑的攻击力数值再加300<br/>equipId: 装备id<br/>valueType: 增幅类型,只能是value(数值)或percentage(百分比)<br/>name: 要修改的属性名称,如atk<br/>value: 要修改到的属性数值<br/>operator: 操作符,可选,如+=表示在原始值上增加<br/>prefix: 独立开关前缀,一般不需要",
-          "!type": "fn(equipId: string, valueType: string, name: string, value: ?, operator?: string, prefix?: string)"
-        }
       },
-      "utils": {
+      utils: {
         "!doc": "工具函数库,里面有各个样板中使用到的工具函数。",
-        "scan": {
+        scan: {
           "!doc": "朝向到x,y映射",
-          "up": {
-            "x": "number",
-            "y": "number"
+          up: {
+            x: "number",
+            y: "number",
           },
-          "down": {
-            "x": "number",
-            "y": "number"
+          down: {
+            x: "number",
+            y: "number",
           },
-          "left": {
-            "x": "number",
-            "y": "number"
+          left: {
+            x: "number",
+            y: "number",
+          },
+          right: {
+            x: "number",
+            y: "number",
           },
-          "right": {
-            "x": "number",
-            "y": "number"
-          }
         },
-        "applyEasing": {
+        applyEasing: {
           "!doc": "获得变速移动曲线",
-          "!type": "fn(mode?: string) -> fn(t: number) -> number"
+          "!type": "fn(mode?: string) -> fn(t: number) -> number",
         },
-        "clamp": {
-          "!doc": "将x限定在[a,b]区间内,注意a和b可交换<br/>例如:core.clamp(1200, 1, 1000); // 1000<br/>x: 原始值,!x为true时x一律视为0<br/>a: 下限值,大于b将导致与b交换<br/>b: 上限值,小于a将导致与a交换",
-          "!type": "fn(x: number, a: number, b: number) -> number"
+        clamp: {
+          "!doc":
+            "将x限定在[a,b]区间内,注意a和b可交换<br/>例如:core.clamp(1200, 1, 1000); // 1000<br/>x: 原始值,!x为true时x一律视为0<br/>a: 下限值,大于b将导致与b交换<br/>b: 上限值,小于a将导致与a交换",
+          "!type": "fn(x: number, a: number, b: number) -> number",
         },
-        "rand": {
-          "!doc": "不支持SL的随机数<br/>例如:1 + core.rand(6); // 随机生成一个小于7的正整数,模拟骰子的效果<br/>num: 填正数表示生成小于num的随机自然数,否则生成小于1的随机正数<br/>返回值:随机数,即使读档也不会改变结果",
-          "!type": "fn(num?: number) -> number"
+        rand: {
+          "!doc":
+            "不支持SL的随机数<br/>例如:1 + core.rand(6); // 随机生成一个小于7的正整数,模拟骰子的效果<br/>num: 填正数表示生成小于num的随机自然数,否则生成小于1的随机正数<br/>返回值:随机数,即使读档也不会改变结果",
+          "!type": "fn(num?: number) -> number",
         },
-        "clone": {
-          "!doc": "深拷贝一个对象(函数将原样返回)<br/>例如:core.clone(core.status.hero, (name, value) => (name == 'items' || typeof value == 'number'), false); // 深拷贝主角的属性和道具<br/>data: 待拷贝对象<br/>filter: 过滤器,可选,表示data为数组或对象时拷贝哪些项或属性,true表示拷贝<br/>recursion: 过滤器是否递归,可选。true表示过滤器也被递归<br/>返回值:拷贝的结果,注意函数将原样返回",
-          "!type": "fn(data?: ?, filter?: fn(name: string, value: ?) -> bool, recursion?: bool)"
+        clone: {
+          "!doc":
+            "深拷贝一个对象(函数将原样返回)<br/>例如:core.clone(core.status.hero, (name, value) => (name == 'items' || typeof value == 'number'), false); // 深拷贝主角的属性和道具<br/>data: 待拷贝对象<br/>filter: 过滤器,可选,表示data为数组或对象时拷贝哪些项或属性,true表示拷贝<br/>recursion: 过滤器是否递归,可选。true表示过滤器也被递归<br/>返回值:拷贝的结果,注意函数将原样返回",
+          "!type":
+            "fn(data?: ?, filter?: fn(name: string, value: ?) -> bool, recursion?: bool)",
         },
-        "cloneArray": {
-          "!doc": "深拷贝一个1D或2D数组对象<br/>例如:core.cloneArray(core.status.thisMap.map)",
-          "!type": "fn(data?: [number]|[[number]]) -> [number]|[[number]]"
+        cloneArray: {
+          "!doc":
+            "深拷贝一个1D或2D数组对象<br/>例如:core.cloneArray(core.status.thisMap.map)",
+          "!type": "fn(data?: [number]|[[number]]) -> [number]|[[number]]",
         },
-        "setLocalForage": {
+        setLocalForage: {
           "!doc": "往数据库写入一段数据",
-          "!type": "fn(key: string, value?: ?, successCallback?: fn(), errorCallback?: fn())"
+          "!type":
+            "fn(key: string, value?: ?, successCallback?: fn(), errorCallback?: fn())",
         },
-        "getGlobal": {
-          "!doc": "读取一个全局存储,适用于global:xxx,支持录像。<br/>例如:if (core.getGlobal('一周目已通关', false) === true) core.getItem('dagger'); // 二周目游戏进行到此处时会获得一把屠龙匕首<br/>key: 全局变量名称,支持中文<br/>defaultValue: 可选,当此全局变量不存在或值为null、undefined时,用此值代替<br/>返回值:全局变量的值",
-          "!type": "fn(key: string, defaultValue?: ?)"
+        getGlobal: {
+          "!doc":
+            "读取一个全局存储,适用于global:xxx,支持录像。<br/>例如:if (core.getGlobal('一周目已通关', false) === true) core.getItem('dagger'); // 二周目游戏进行到此处时会获得一把屠龙匕首<br/>key: 全局变量名称,支持中文<br/>defaultValue: 可选,当此全局变量不存在或值为null、undefined时,用此值代替<br/>返回值:全局变量的值",
+          "!type": "fn(key: string, defaultValue?: ?)",
         },
-        "replaceText": {
-          "!doc": "将一段文字中的${}(表达式)进行替换。<br/>例如:core.replaceText('衬衫的价格是${status:hp}镑${item:yellowKey}便士。'); // 把主角的生命值和持有的黄钥匙数量代入这句话<br/>text: 模板字符串,可以使用${}计算js表达式,支持“状态、物品、变量、独立开关、全局存储、图块id、图块类型、敌人数据、装备id”等量参与运算<br/>返回值:替换完毕后的字符串",
-          "!type": "fn(text: string, prefix?: string) -> string"
+        replaceText: {
+          "!doc":
+            "将一段文字中的${}(表达式)进行替换。<br/>例如:core.replaceText('衬衫的价格是${status:hp}镑${item:yellowKey}便士。'); // 把主角的生命值和持有的黄钥匙数量代入这句话<br/>text: 模板字符串,可以使用${}计算js表达式,支持“状态、物品、变量、独立开关、全局存储、图块id、图块类型、敌人数据、装备id”等量参与运算<br/>返回值:替换完毕后的字符串",
+          "!type": "fn(text: string, prefix?: string) -> string",
         },
-        "removeLocalStorage": {
+        removeLocalStorage: {
           "!doc": "移除本地存储",
-          "!type": "fn(key: string)"
+          "!type": "fn(key: string)",
         },
-        "unzip": {
+        unzip: {
           "!doc": "解压一段内容",
-          "!type": "fn(blobOrUrl?: ?, success?: fn(data: ?), error?: fn(error: string), convertToText?: bool, onprogress?: fn(loaded: number, total: number))"
+          "!type":
+            "fn(blobOrUrl?: ?, success?: fn(data: ?), error?: fn(error: string), convertToText?: bool, onprogress?: fn(loaded: number, total: number))",
         },
-        "formatTime": {
+        formatTime: {
           "!doc": "格式化时间",
-          "!type": "fn(time: number) -> string"
+          "!type": "fn(time: number) -> string",
         },
-        "readFile": {
-          "!doc": "尝试请求读取一个本地文件内容 [异步]<br/>success: 成功后的回调<br/>error: 失败后的回调<br/>readType: 不设置则以文本读取,否则以DataUrl形式读取",
-          "!type": "fn(success?: fn(data: string), error?: fn(message: string), readType?: bool)"
+        readFile: {
+          "!doc":
+            "尝试请求读取一个本地文件内容 [异步]<br/>success: 成功后的回调<br/>error: 失败后的回调<br/>readType: 不设置则以文本读取,否则以DataUrl形式读取",
+          "!type":
+            "fn(success?: fn(data: string), error?: fn(message: string), readType?: bool)",
         },
-        "readFileContent": {
+        readFileContent: {
           "!doc": "文件读取完毕后的内容处理 [异步]",
-          "!type": "fn(content: string)"
+          "!type": "fn(content: string)",
         },
-        "formatDate": {
+        formatDate: {
           "!doc": "格式化日期为字符串",
-          "!type": "fn(date: ?) -> string"
+          "!type": "fn(date: ?) -> string",
         },
-        "download": {
-          "!doc": "弹窗请求下载一个文本文件<br/>例如:core.download('route.txt', JSON.stringify(core.status.route)); // 弹窗请求下载录像<br/>filename: 文件名<br/>content: 文件内容",
-          "!type": "fn(filename: string, content: string)"
+        download: {
+          "!doc":
+            "弹窗请求下载一个文本文件<br/>例如:core.download('route.txt', JSON.stringify(core.status.route)); // 弹窗请求下载录像<br/>filename: 文件名<br/>content: 文件内容",
+          "!type": "fn(filename: string, content: string)",
         },
-        "encodeBase64": {
-          "!doc": "base64加密<br/>例如:core.encodeBase64('abcd'); // 'YWJjZA=='<br/>str: 明文<br/>返回值:密文",
-          "!type": "fn(str: string) -> string"
+        encodeBase64: {
+          "!doc":
+            "base64加密<br/>例如:core.encodeBase64('abcd'); // 'YWJjZA=='<br/>str: 明文<br/>返回值:密文",
+          "!type": "fn(str: string) -> string",
         },
-        "strlen": {
-          "!doc": "求字符串的国标码字节数,也可用于等宽字体下文本的宽度测算。请注意样板的默认字体Verdana不是等宽字体<br/>例如:core.strlen('无敌ad'); // 6<br/>str: 待测字符串<br/>返回值:字符串的国标码字节数,每个汉字为2,每个ASCII字符为1",
-          "!type": "fn(str: string) -> number"
+        strlen: {
+          "!doc":
+            "求字符串的国标码字节数,也可用于等宽字体下文本的宽度测算。请注意样板的默认字体Verdana不是等宽字体<br/>例如:core.strlen('无敌ad'); // 6<br/>str: 待测字符串<br/>返回值:字符串的国标码字节数,每个汉字为2,每个ASCII字符为1",
+          "!type": "fn(str: string) -> number",
         },
-        "myprompt": {
+        myprompt: {
           "!doc": "让用户输入一段文字",
-          "!type": "fn(hint: string, value: string, callback?: fn(data?: string))"
+          "!type":
+            "fn(hint: string, value: string, callback?: fn(data?: string))",
         },
-        "getCookie": {
+        getCookie: {
           "!doc": "访问浏览器cookie",
-          "!type": "fn(name: string) -> string"
+          "!type": "fn(name: string) -> string",
         },
-        "decodeRoute": {
-          "!doc": "录像解压的最后一步,即一压的逆过程<br/>例如:core.decodeRoute(core.encodeRoute(core.status.route)); // 一压当前录像再解压-_-|<br/>route: 录像解压倒数第二步的结果,即一压的结果<br/>返回值:原始录像",
-          "!type": "fn(route: string) -> [string]"
+        decodeRoute: {
+          "!doc":
+            "录像解压的最后一步,即一压的逆过程<br/>例如:core.decodeRoute(core.encodeRoute(core.status.route)); // 一压当前录像再解压-_-|<br/>route: 录像解压倒数第二步的结果,即一压的结果<br/>返回值:原始录像",
+          "!type": "fn(route: string) -> [string]",
         },
-        "formatDate2": {
+        formatDate2: {
           "!doc": "格式化日期为最简字符串",
-          "!type": "fn(date: ?) -> string"
+          "!type": "fn(date: ?) -> string",
         },
-        "unshift": {
-          "!doc": "将b(可以是另一个数组)插入数组a的开头,此函数用于弥补a.unshift(b)中b只能是单项的不足。<br/>例如:core.unshift(todo, {type: 'unfollow'}); // 在事件指令数组todo的开头插入“取消所有跟随者”指令<br/>a: 原数组<br/>b: 待插入的新首项或前缀数组<br/>返回值:插入完毕后的新数组,它是改变原数组a本身得到的",
-          "!type": "fn(a: [?], b: ?) -> [?]"
+        unshift: {
+          "!doc":
+            "将b(可以是另一个数组)插入数组a的开头,此函数用于弥补a.unshift(b)中b只能是单项的不足。<br/>例如:core.unshift(todo, {type: 'unfollow'}); // 在事件指令数组todo的开头插入“取消所有跟随者”指令<br/>a: 原数组<br/>b: 待插入的新首项或前缀数组<br/>返回值:插入完毕后的新数组,它是改变原数组a本身得到的",
+          "!type": "fn(a: [?], b: ?) -> [?]",
         },
-        "same": {
-          "!doc": "判定深层相等, 会逐层比较每个元素<br/>例如:core.same(['1', 2], ['1', 2]); // true",
-          "!type": "fn(a?: ?, b?: ?) -> bool"
+        same: {
+          "!doc":
+            "判定深层相等, 会逐层比较每个元素<br/>例如:core.same(['1', 2], ['1', 2]); // true",
+          "!type": "fn(a?: ?, b?: ?) -> bool",
         },
-        "setTwoDigits": {
+        setTwoDigits: {
           "!doc": "两位数显示",
-          "!type": "fn(x: number) -> string"
+          "!type": "fn(x: number) -> string",
         },
-        "splitImage": {
-          "!doc": "等比例切分一张图片<br/>例如:core.splitImage(core.material.images.images['npc48.png'], 32, 48); // 把npc48.png切分成若干32×48px的小人<br/>image: 图片名(支持映射前的中文名)或图片对象(参见上面的例子),获取不到时返回[]<br/>width: 子图的宽度,单位为像素。原图总宽度必须是其倍数,不填视为32<br/>height: 子图的高度,单位为像素。原图总高度必须是其倍数,不填视为正方形<br/>返回值:子图组成的数组,在原图中呈先行后列,从左到右、从上到下排列。",
-          "!type": "fn(image?: string|image, width?: number, height?: number) -> [image]"
+        splitImage: {
+          "!doc":
+            "等比例切分一张图片<br/>例如:core.splitImage(core.material.images.images['npc48.png'], 32, 48); // 把npc48.png切分成若干32×48px的小人<br/>image: 图片名(支持映射前的中文名)或图片对象(参见上面的例子),获取不到时返回[]<br/>width: 子图的宽度,单位为像素。原图总宽度必须是其倍数,不填视为32<br/>height: 子图的高度,单位为像素。原图总高度必须是其倍数,不填视为正方形<br/>返回值:子图组成的数组,在原图中呈先行后列,从左到右、从上到下排列。",
+          "!type":
+            "fn(image?: string|image, width?: number, height?: number) -> [image]",
         },
-        "decompress": {
+        decompress: {
           "!doc": "解压缩一个数据",
-          "!type": "fn(value: ?)"
+          "!type": "fn(value: ?)",
         },
-        "showWithAnimate": {
+        showWithAnimate: {
           "!doc": "动画显示某对象",
-          "!type": "fn(obj?: ?, speed?: number, callback?: fn())"
+          "!type": "fn(obj?: ?, speed?: number, callback?: fn())",
         },
-        "subarray": {
-          "!doc": "判定一个数组是否为另一个数组的前缀,用于录像接续播放。请注意函数名没有大写字母<br/>例如:core.subarray(['ad', '米库', '小精灵', '小破草', '小艾'], ['ad', '米库', '小精灵']); // ['小破草', '小艾']<br/>a: 可能的母数组,不填或比b短将返回null<br/>b: 可能的前缀,不填或比a长将返回null<br/>返回值:如果b不是a的前缀将返回null,否则将返回a去掉此前缀后的剩余数组",
-          "!type": "fn(a?: [?], b?: [?]) -> [?]|null"
+        subarray: {
+          "!doc":
+            "判定一个数组是否为另一个数组的前缀,用于录像接续播放。请注意函数名没有大写字母<br/>例如:core.subarray(['ad', '米库', '小精灵', '小破草', '小艾'], ['ad', '米库', '小精灵']); // ['小破草', '小艾']<br/>a: 可能的母数组,不填或比b短将返回null<br/>b: 可能的前缀,不填或比a长将返回null<br/>返回值:如果b不是a的前缀将返回null,否则将返回a去掉此前缀后的剩余数组",
+          "!type": "fn(a?: [?], b?: [?]) -> [?]|null",
         },
-        "turnDirection": {
-          "!doc": "计算应当转向某个方向<br/>turn: 转向的方向,可为 up,down,left,right,:left,:right,:back 七种<br/>direction: 当前方向",
-          "!type": "fn(turn: string, direction?: string) -> string"
+        turnDirection: {
+          "!doc":
+            "计算应当转向某个方向<br/>turn: 转向的方向,可为 up,down,left,right,:left,:right,:back 七种<br/>direction: 当前方向",
+          "!type": "fn(turn: string, direction?: string) -> string",
         },
-        "myconfirm": {
-          "!doc": "显示确认框,类似core.drawConfirmBox(),但不打断事件流<br/>例如:core.myconfirm('重启游戏?', core.restart); // 弹窗询问玩家是否重启游戏<br/>hint: 弹窗的内容,支持 ${} 语法<br/>yesCallback: 确定后的回调函数<br/>noCallback: 取消后的回调函数,可选",
-          "!type": "fn(hint: string, yesCallback?: fn(), noCallback?: fn())"
+        myconfirm: {
+          "!doc":
+            "显示确认框,类似core.drawConfirmBox(),但不打断事件流<br/>例如:core.myconfirm('重启游戏?', core.restart); // 弹窗询问玩家是否重启游戏<br/>hint: 弹窗的内容,支持 ${} 语法<br/>yesCallback: 确定后的回调函数<br/>noCallback: 取消后的回调函数,可选",
+          "!type": "fn(hint: string, yesCallback?: fn(), noCallback?: fn())",
         },
-        "calValue": {
-          "!doc": "计算一个表达式的值,支持status:xxx等的计算。<br/>例如:core.calValue('status:hp + status:def'); // 计算主角的生命值加防御力<br/>value: 待求值的表达式<br/>prefix: 独立开关前缀,一般可省略<br/>返回值:求出的值",
-          "!type": "fn(value: string, prefix?: string)"
+        calValue: {
+          "!doc":
+            "计算一个表达式的值,支持status:xxx等的计算。<br/>例如:core.calValue('status:hp + status:def'); // 计算主角的生命值加防御力<br/>value: 待求值的表达式<br/>prefix: 独立开关前缀,一般可省略<br/>返回值:求出的值",
+          "!type": "fn(value: string, prefix?: string)",
         },
-        "encodeRoute": {
-          "!doc": "录像压缩缩<br/>例如:core.encodeRoute(core.status.route); // 压缩当前录像<br/>route: 原始录像,自定义内容(不予压缩,原样写入)必须由0-9A-Za-z和下划线、冒号组成,所以中文和数组需要用JSON.stringify预处理再base64压缩才能交由一压<br/>返回值:一压的结果",
-          "!type": "fn(route: [string]) -> string"
+        encodeRoute: {
+          "!doc":
+            "录像压缩缩<br/>例如:core.encodeRoute(core.status.route); // 压缩当前录像<br/>route: 原始录像,自定义内容(不予压缩,原样写入)必须由0-9A-Za-z和下划线、冒号组成,所以中文和数组需要用JSON.stringify预处理再base64压缩才能交由一压<br/>返回值:一压的结果",
+          "!type": "fn(route: [string]) -> string",
         },
-        "decodeBase64": {
-          "!doc": "base64解密<br/>例如:core.decodeBase64('YWJjZA=='); // \"abcd\"<br/>str: 密文<br/>返回值:明文",
-          "!type": "fn(str: string) -> string"
+        decodeBase64: {
+          "!doc":
+            "base64解密<br/>例如:core.decodeBase64('YWJjZA=='); // \"abcd\"<br/>str: 密文<br/>返回值:明文",
+          "!type": "fn(str: string) -> string",
         },
-        "http": {
-          "!doc": "发送一个HTTP请求 [异步]<br/>type: 请求类型,只能为GET或POST<br/>url: 目标地址<br/>formData: 如果是POST请求则为表单数据<br/>success: 成功后的回调<br/>error: 失败后的回调",
-          "!type": "fn(type: string, url: string, formData: ?, success?: fn(data: string), error?: fn(message: string), mimeType?: string, responseType?: string, onprogress?: fn(loaded: number, total: number))"
+        http: {
+          "!doc":
+            "发送一个HTTP请求 [异步]<br/>type: 请求类型,只能为GET或POST<br/>url: 目标地址<br/>formData: 如果是POST请求则为表单数据<br/>success: 成功后的回调<br/>error: 失败后的回调",
+          "!type":
+            "fn(type: string, url: string, formData: ?, success?: fn(data: string), error?: fn(message: string), mimeType?: string, responseType?: string, onprogress?: fn(loaded: number, total: number))",
         },
-        "getGuid": {
+        getGuid: {
           "!doc": "获得或生成浏览器唯一的guid",
-          "!type": "fn() -> string"
+          "!type": "fn() -> string",
         },
-        "getLocalStorage": {
+        getLocalStorage: {
           "!doc": "获得本地存储",
-          "!type": "fn(key: string, defaultValue?: ?)"
+          "!type": "fn(key: string, defaultValue?: ?)",
         },
-        "arrayToRGB": {
-          "!doc": "颜色数组转字符串<br/>例如:core.arrayToRGB([102, 204, 255]); // \"#66ccff\"<br/>color: 一行三列的数组,必须为不大于255的自然数<br/>返回值:该颜色的#xxxxxx字符串表示",
-          "!type": "fn(color: [number]) -> string"
+        arrayToRGB: {
+          "!doc":
+            '颜色数组转字符串<br/>例如:core.arrayToRGB([102, 204, 255]); // "#66ccff"<br/>color: 一行三列的数组,必须为不大于255的自然数<br/>返回值:该颜色的#xxxxxx字符串表示',
+          "!type": "fn(color: [number]) -> string",
         },
-        "arrayToRGBA": {
-          "!doc": "颜色数组转字符串<br/>例如:core.arrayToRGBA([102, 204, 255, 0.3]); // \"rgba(102,204,255,0.3)\"<br/>color: 一行三列或一行四列的数组,前三个元素必须为不大于255的自然数。第四个元素(如果有)必须为0或不大于1的数字,第四个元素不填视为1<br/>返回值:该颜色的rgba(...)字符串表示",
-          "!type": "fn(color: [number]) -> string"
+        arrayToRGBA: {
+          "!doc":
+            '颜色数组转字符串<br/>例如:core.arrayToRGBA([102, 204, 255, 0.3]); // "rgba(102,204,255,0.3)"<br/>color: 一行三列或一行四列的数组,前三个元素必须为不大于255的自然数。第四个元素(如果有)必须为0或不大于1的数字,第四个元素不填视为1<br/>返回值:该颜色的rgba(...)字符串表示',
+          "!type": "fn(color: [number]) -> string",
         },
-        "formatBigNumber": {
-          "!doc": "大数字格式化,单位为10000的倍数(w,e,z,j,g),末尾四舍五入<br/>例如:core.formatBigNumber(123456789, false); // \"12346w\"<br/>x: 原数字<br/>onMap: 可选,true表示用于地图显伤,结果总字符数最多为5,否则最多为6<br/>返回值:格式化结果",
-          "!type": "fn(x: number, onMap?: bool) -> string"
+        formatBigNumber: {
+          "!doc":
+            '大数字格式化,单位为10000的倍数(w,e,z,j,g),末尾四舍五入<br/>例如:core.formatBigNumber(123456789, false); // "12346w"<br/>x: 原数字<br/>onMap: 可选,true表示用于地图显伤,结果总字符数最多为5,否则最多为6<br/>返回值:格式化结果',
+          "!type": "fn(x: number, onMap?: bool) -> string",
         },
-        "removeLocalForage": {
+        removeLocalForage: {
           "!doc": "移除本地数据库的数据",
-          "!type": "fn(key: string, successCallback?: fn(), errorCallback?: fn())"
+          "!type":
+            "fn(key: string, successCallback?: fn(), errorCallback?: fn())",
         },
-        "matchWildcard": {
-          "!doc": "通配符匹配,用于搜索图块等批量处理。<br/>例如:core.playSound(core.matchWildcard('*Key', itemId) ? 'item.mp3' : 'door.mp3'); // 判断捡到的是钥匙还是别的道具,从而播放不同的音效<br/>pattern: 模式串,每个星号表示任意多个(0个起)字符<br/>string: 待测串<br/>返回值:true表示匹配成功,false表示匹配失败",
-          "!type": "fn(pattern: string, string: string) -> bool"
+        matchWildcard: {
+          "!doc":
+            "通配符匹配,用于搜索图块等批量处理。<br/>例如:core.playSound(core.matchWildcard('*Key', itemId) ? 'item.opus' : 'door.opus'); // 判断捡到的是钥匙还是别的道具,从而播放不同的音效<br/>pattern: 模式串,每个星号表示任意多个(0个起)字符<br/>string: 待测串<br/>返回值:true表示匹配成功,false表示匹配失败",
+          "!type": "fn(pattern: string, string: string) -> bool",
         },
-        "setLocalStorage": {
+        setLocalStorage: {
           "!doc": "设置本地存储",
-          "!type": "fn(key: string, value?: ?)"
+          "!type": "fn(key: string, value?: ?)",
         },
-        "hideWithAnimate": {
+        hideWithAnimate: {
           "!doc": "动画使某对象消失",
-          "!type": "fn(obj?: ?, speed?: number, callback?: fn())"
+          "!type": "fn(obj?: ?, speed?: number, callback?: fn())",
         },
-        "copy": {
+        copy: {
           "!doc": "尝试复制一段文本到剪切板。",
-          "!type": "fn(data: string) -> bool"
+          "!type": "fn(data: string) -> bool",
         },
-        "isset": {
-          "!doc": "判断一个值是否不为null,undefined和NaN<br/>例如:core.isset(0/0); // false,因为0/0等于NaN<br/>v: 待测值,可选<br/>返回值:false表示待测值为null、undefined、NaN或未填写,true表示为其他值。",
-          "!type": "fn(v?: ?) -> bool"
+        isset: {
+          "!doc":
+            "判断一个值是否不为null,undefined和NaN<br/>例如:core.isset(0/0); // false,因为0/0等于NaN<br/>v: 待测值,可选<br/>返回值:false表示待测值为null、undefined、NaN或未填写,true表示为其他值。",
+          "!type": "fn(v?: ?) -> bool",
         },
-        "replaceValue": {
-          "!doc": "对一个表达式中的特殊规则进行替换,如status:xxx等。<br/>例如:core.replaceValue('status:atk+item:yellowKey'); // 把这两个冒号表达式替换为core.getStatus('hp')和core.itemCount('yellowKey')这样的函数调用<br/>value: 模板字符串,注意独立开关不会被替换<br/>返回值:替换完毕后的字符串",
-          "!type": "fn(value: string) -> string"
+        replaceValue: {
+          "!doc":
+            "对一个表达式中的特殊规则进行替换,如status:xxx等。<br/>例如:core.replaceValue('status:atk+item:yellowKey'); // 把这两个冒号表达式替换为core.getStatus('hp')和core.itemCount('yellowKey')这样的函数调用<br/>value: 模板字符串,注意独立开关不会被替换<br/>返回值:替换完毕后的字符串",
+          "!type": "fn(value: string) -> string",
         },
-        "getLocalForage": {
+        getLocalForage: {
           "!doc": "从本地数据库读出一段数据",
-          "!type": "fn(key: string, defaultValue?: ?, successCallback?: fn(data: ?), errorCallback?: fn())"
+          "!type":
+            "fn(key: string, defaultValue?: ?, successCallback?: fn(data: ?), errorCallback?: fn())",
         },
-        "inArray": {
-          "!doc": "判定array是不是一个数组,以及element是否在该数组中。<br/>array: 可能的数组,不为数组或不填将导致返回值为false<br/>element: 待查找的元素<br/>返回值:如果array为数组且具有element这项,就返回true,否则返回false",
-          "!type": "fn(array?: ?, element?: ?) -> bool"
+        inArray: {
+          "!doc":
+            "判定array是不是一个数组,以及element是否在该数组中。<br/>array: 可能的数组,不为数组或不填将导致返回值为false<br/>element: 待查找的元素<br/>返回值:如果array为数组且具有element这项,就返回true,否则返回false",
+          "!type": "fn(array?: ?, element?: ?) -> bool",
         },
-        "setGlobal": {
-          "!doc": "设置一个全局存储,适用于global:xxx,录像播放时将忽略此函数。<br/>例如:core.setBlobal('一周目已通关', true); // 设置全局存储“一周目已通关”为true,方便二周目游戏中的新要素。<br/>key: 全局变量名称,支持中文<br/>value: 全局变量的新值,不填或null表示清除此全局存储",
-          "!type": "fn(key: string, value?: ?)"
+        setGlobal: {
+          "!doc":
+            "设置一个全局存储,适用于global:xxx,录像播放时将忽略此函数。<br/>例如:core.setBlobal('一周目已通关', true); // 设置全局存储“一周目已通关”为true,方便二周目游戏中的新要素。<br/>key: 全局变量名称,支持中文<br/>value: 全局变量的新值,不填或null表示清除此全局存储",
+          "!type": "fn(key: string, value?: ?)",
         },
-        "rand2": {
-          "!doc": "支持SL的随机数,并计入录像<br/>例如:1 + core.rand2(6); // 随机生成一个小于7的正整数,模拟骰子的效果<br/>num: 正整数,0或不填会被视为2147483648<br/>返回值:属于 [0, num) 的随机数",
-          "!type": "fn(num?: number) -> number"
+        rand2: {
+          "!doc":
+            "支持SL的随机数,并计入录像<br/>例如:1 + core.rand2(6); // 随机生成一个小于7的正整数,模拟骰子的效果<br/>num: 正整数,0或不填会被视为2147483648<br/>返回值:属于 [0, num) 的随机数",
+          "!type": "fn(num?: number) -> number",
         },
-        "setStatusBarInnerHTML": {
-          "!doc": "填写非自绘状态栏<br/>例如:core.setStatusBarInnerHTML('hp', core.status.hero.hp, 'color: #66CCFF'); // 更新状态栏中的主角生命,使用加载画面的宣传色<br/>name: 状态栏项的名称,如'hp', 'atk', 'def'等。必须是core.statusBar中的一个合法项<br/>value: 要填写的内容,大数字会被格式化为至多6个字符,无中文的内容会被自动设为斜体<br/>css: 额外的css样式,可选。如更改颜色等",
-          "!type": "fn(name: string, value: ?, css?: string)"
+        setStatusBarInnerHTML: {
+          "!doc":
+            "填写非自绘状态栏<br/>例如:core.setStatusBarInnerHTML('hp', core.status.hero.hp, 'color: #66CCFF'); // 更新状态栏中的主角生命,使用加载画面的宣传色<br/>name: 状态栏项的名称,如'hp', 'atk', 'def'等。必须是core.statusBar中的一个合法项<br/>value: 要填写的内容,大数字会被格式化为至多6个字符,无中文的内容会被自动设为斜体<br/>css: 额外的css样式,可选。如更改颜色等",
+          "!type": "fn(name: string, value: ?, css?: string)",
         },
-        "matchRegex": {
+        matchRegex: {
           "!doc": "是否满足正则表达式",
-          "!type": "fn(pattern: string, string: string) -> string"
+          "!type": "fn(pattern: string, string: string) -> string",
         },
-        "push": {
-          "!doc": "将b(可以是另一个数组)插入数组a的末尾,此函数用于弥补a.push(b)中b只能是单项的不足。<br/>例如:core.push(todo, {type: 'unfollow'}); // 在事件指令数组todo的末尾插入“取消所有跟随者”指令<br/>a: 原数组<br/>b: 待插入的新末项或后缀数组<br/>返回值:插入完毕后的新数组,它是改变原数组a本身得到的",
-          "!type": "fn(a: [?], b: ?) -> [?]"
+        push: {
+          "!doc":
+            "将b(可以是另一个数组)插入数组a的末尾,此函数用于弥补a.push(b)中b只能是单项的不足。<br/>例如:core.push(todo, {type: 'unfollow'}); // 在事件指令数组todo的末尾插入“取消所有跟随者”指令<br/>a: 原数组<br/>b: 待插入的新末项或后缀数组<br/>返回值:插入完毕后的新数组,它是改变原数组a本身得到的",
+          "!type": "fn(a: [?], b: ?) -> [?]",
         },
-        "formatSize": {
+        formatSize: {
           "!doc": "格式化文件大小",
-          "!type": "fn(size: number) -> string"
-        }
+          "!type": "fn(size: number) -> string",
+        },
       },
-      "actions": {
+      actions: {
         "!doc": "主要是处理一些和用户交互相关的内容。",
-        "onup": {
+        onup: {
           "!doc": "当点击(触摸)事件放开时",
-          "!type": "fn(loc: {x: number, y: number, size: number})"
+          "!type": "fn(loc: {x: number, y: number, size: number})",
         },
-        "pressKey": {
+        pressKey: {
           "!doc": "按住某个键时",
-          "!type": "fn(keyCode: number)"
+          "!type": "fn(keyCode: number)",
         },
-        "keyUp": {
+        keyUp: {
           "!doc": "根据放开键的code来执行一系列操作",
-          "!type": "fn(keyCode: number, altKey?: bool, fromReplay?: bool)"
+          "!type": "fn(keyCode: number, altKey?: bool, fromReplay?: bool)",
         },
-        "ondown": {
+        ondown: {
           "!doc": "点击(触摸)事件按下时",
-          "!type": "fn(loc: {x: number, y: number, size: number})"
+          "!type": "fn(loc: {x: number, y: number, size: number})",
         },
-        "registerAction": {
-          "!doc": "此函数将注册一个用户交互行为。<br/>action: 要注册的交互类型,如 ondown, onclick, keyDown 等等。<br/>name: 你的自定义名称,可被注销使用;同名重复注册将后者覆盖前者。<br/>func: 执行函数。<br/>如果func返回true,则不会再继续执行其他的交互函数;否则会继续执行其他的交互函数。<br/>priority: 优先级;优先级高的将会被执行。此项可不填,默认为0",
-          "!type": "fn(action: string, name: string, func: string|fn(params: ?), priority?: number)"
+        registerAction: {
+          "!doc":
+            "此函数将注册一个用户交互行为。<br/>action: 要注册的交互类型,如 ondown, onclick, keyDown 等等。<br/>name: 你的自定义名称,可被注销使用;同名重复注册将后者覆盖前者。<br/>func: 执行函数。<br/>如果func返回true,则不会再继续执行其他的交互函数;否则会继续执行其他的交互函数。<br/>priority: 优先级;优先级高的将会被执行。此项可不填,默认为0",
+          "!type":
+            "fn(action: string, name: string, func: string|fn(params: ?), priority?: number)",
         },
-        "onkeyDown": {
+        onkeyDown: {
           "!doc": "按下某个键时",
-          "!type": "fn(e: Event)"
+          "!type": "fn(e: Event)",
         },
-        "keyDown": {
+        keyDown: {
           "!doc": "根据按下键的code来执行一系列操作",
-          "!type": "fn(keyCode: number)"
+          "!type": "fn(keyCode: number)",
         },
-        "onStatusBarClick": {
+        onStatusBarClick: {
           "!doc": "点击自绘状态栏时",
-          "!type": "fn(e?: Event)"
+          "!type": "fn(e?: Event)",
         },
-        "longClick": {
+        longClick: {
           "!doc": "长按",
-          "!type": "fn(x: number, y: number, px: number, py: number, fromEvent?: bool)"
+          "!type":
+            "fn(x: number, y: number, px: number, py: number, fromEvent?: bool)",
         },
-        "unregisterAction": {
+        unregisterAction: {
           "!doc": "注销一个用户交互行为",
-          "!type": "fn(action: string, name: string)"
+          "!type": "fn(action: string, name: string)",
         },
-        "keyDownCtrl": {
+        keyDownCtrl: {
           "!doc": "长按Ctrl键时",
-          "!type": "fn() -> bool"
+          "!type": "fn() -> bool",
         },
-        "onclick": {
+        onclick: {
           "!doc": "具体点击屏幕上(x,y)点时,执行的操作",
-          "!type": "fn(x: number, y: number, px: number, py: number, stepPostfix?: [?])"
+          "!type":
+            "fn(x: number, y: number, px: number, py: number, stepPostfix?: [?])",
         },
-        "doRegisteredAction": {
+        doRegisteredAction: {
           "!doc": "执行一个用户交互行为",
-          "!type": "fn(action: string, params: ?)"
+          "!type": "fn(action: string, params: ?)",
         },
-        "onkeyUp": {
+        onkeyUp: {
           "!doc": "放开某个键时",
-          "!type": "fn(e: Event)"
+          "!type": "fn(e: Event)",
         },
-        "onmousewheel": {
+        onmousewheel: {
           "!doc": "滑动鼠标滚轮时的操作",
-          "!type": "fn(direct: number)"
+          "!type": "fn(direct: number)",
         },
-        "onmove": {
+        onmove: {
           "!doc": "当在触摸屏上滑动时",
-          "!type": "fn(loc: {x: number, y: number, size: number})"
-        }
+          "!type": "fn(loc: {x: number, y: number, size: number})",
+        },
       },
-      "loader": {
+      loader: {
         "!doc": "资源加载相关的函数",
-        "loadImages": {
+        loadImages: {
           "!doc": "加载一系列图片",
-          "!type": "fn(dir: string, names: [string], toSave: ?, callback?: fn()) "
+          "!type":
+            "fn(dir: string, names: [string], toSave: ?, callback?: fn()) ",
         },
-        "loadImagesFromZip": {
+        loadImagesFromZip: {
           "!doc": "从zip中加载一系列图片",
-          "!type": "fn(url: string, names: [string], toSave?: ?, onprogress?: ?, onfinished?: ?)"
+          "!type":
+            "fn(url: string, names: [string], toSave?: ?, onprogress?: ?, onfinished?: ?)",
         },
-        "loadBgm": {
+        loadBgm: {
           "!doc": "加载一个bgm",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "loadOneMusic": {
+        loadOneMusic: {
           "!doc": "加载一个音乐或音效",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "freeBgm": {
+        freeBgm: {
           "!doc": "释放一个bgm的缓存",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "loadOneSound": {
+        loadOneSound: {
           "!doc": "加载一个音效",
-          "!type": "fn(name: string)"
+          "!type": "fn(name: string)",
         },
-        "loadImage": {
+        loadImage: {
           "!doc": "加载某一张图片",
-          "!type": "fn(dir: name, imgName: name, callback?: fn())"
-        }
+          "!type": "fn(dir: name, imgName: name, callback?: fn())",
+        },
       },
-      "maps": {
-        "!doc": "负责一切和地图相关的处理内容,包括如下几个方面:<br/>- 地图的初始化,保存和读取,地图数组的生成<br/>- 是否可移动或瞬间移动的判定<br/>- 地图的绘制<br/>- 获得某个点的图块信息<br/>- 启用和禁用图块,改变图块	<br/>- 移动/跳跃图块,淡入淡出图块<br/>- 全局动画控制,动画的绘制",
-        "noPass": {
-          "!doc": "判定某个点是否不可被踏入(不基于主角生命值和图块cannotIn属性)<br/>例如:core.noPass(0, 0); // 判断地图左上角能否被踏入<br/>x: 目标点的横坐标<br/>y: 目标点的纵坐标<br/>floorId: 目标点所在的地图id,不填视为当前地图<br/>返回值:true表示可踏入",
-          "!type": "fn(x: number, y: number, floorId?: string) -> bool"
+      maps: {
+        "!doc":
+          "负责一切和地图相关的处理内容,包括如下几个方面:<br/>- 地图的初始化,保存和读取,地图数组的生成<br/>- 是否可移动或瞬间移动的判定<br/>- 地图的绘制<br/>- 获得某个点的图块信息<br/>- 启用和禁用图块,改变图块	<br/>- 移动/跳跃图块,淡入淡出图块<br/>- 全局动画控制,动画的绘制",
+        noPass: {
+          "!doc":
+            "判定某个点是否不可被踏入(不基于主角生命值和图块cannotIn属性)<br/>例如:core.noPass(0, 0); // 判断地图左上角能否被踏入<br/>x: 目标点的横坐标<br/>y: 目标点的纵坐标<br/>floorId: 目标点所在的地图id,不填视为当前地图<br/>返回值:true表示可踏入",
+          "!type": "fn(x: number, y: number, floorId?: string) -> bool",
         },
-        "drawAnimate": {
-          "!doc": "播放动画,注意即使指定了主角的坐标也不会跟随主角移动,如有需要请使用core.drawHeroAnimate(name, callback)函数<br/>例如:core.drawAnimate('attack', core.nextX(), core.nextY(), false, core.vibrate); // 在主角面前一格播放普攻动画,动画停止后视野左右抖动1秒<br/>name: 动画文件名,不含后缀<br/>x: 横坐标<br/>y: 纵坐标<br/>alignWindow: 是否是相对窗口的坐标<br/>callback: 动画停止后的回调函数,可选<br/>返回值:一个数字,可作为core.stopAnimate()的参数来立即停止播放(届时还可选择是否执行此次播放的回调函数)",
-          "!type": "fn(name: string, x: number, y: number, alignWindow: bool, callback?: fn()) -> number"
+        drawAnimate: {
+          "!doc":
+            "播放动画,注意即使指定了主角的坐标也不会跟随主角移动,如有需要请使用core.drawHeroAnimate(name, callback)函数<br/>例如:core.drawAnimate('attack', core.nextX(), core.nextY(), false, core.vibrate); // 在主角面前一格播放普攻动画,动画停止后视野左右抖动1秒<br/>name: 动画文件名,不含后缀<br/>x: 横坐标<br/>y: 纵坐标<br/>alignWindow: 是否是相对窗口的坐标<br/>callback: 动画停止后的回调函数,可选<br/>返回值:一个数字,可作为core.stopAnimate()的参数来立即停止播放(届时还可选择是否执行此次播放的回调函数)",
+          "!type":
+            "fn(name: string, x: number, y: number, alignWindow: bool, callback?: fn()) -> number",
         },
-        "drawHeroAnimate": {
-          "!doc": "播放跟随勇士的动画<br/>name: 动画名<br/>callback: 动画停止后的回调函数,可选<br/>返回值:一个数字,可作为core.stopAnimate()的参数来立即停止播放(届时还可选择是否执行此次播放的回调函数)",
-          "!type": "fn(name: string, callback?: fn()) -> number"
+        drawHeroAnimate: {
+          "!doc":
+            "播放跟随勇士的动画<br/>name: 动画名<br/>callback: 动画停止后的回调函数,可选<br/>返回值:一个数字,可作为core.stopAnimate()的参数来立即停止播放(届时还可选择是否执行此次播放的回调函数)",
+          "!type": "fn(name: string, callback?: fn()) -> number",
         },
-        "stopAnimate": {
-          "!doc": "立刻停止一个动画播放<br/>id: 播放动画的编号,即drawAnimate或drawHeroAnimate的返回值;不填视为所有动画br/>doCallback: 是否执行该动画的回调函数",
-          "!type": "fn(id?: number, doCallback?: bool)"
+        stopAnimate: {
+          "!doc":
+            "立刻停止一个动画播放<br/>id: 播放动画的编号,即drawAnimate或drawHeroAnimate的返回值;不填视为所有动画br/>doCallback: 是否执行该动画的回调函数",
+          "!type": "fn(id?: number, doCallback?: bool)",
         },
-        "getPlayingAnimates": {
-          "!doc": "获得当前正在播放的所有(指定)动画的id列表<br/>name: 动画名;不填代表返回全部正在播放的动画<br/>返回值: 一个数组,每一项为一个正在播放的动画;可用core.stopAnimate停止播放。",
-          "!type": "fn(name?: string) -> [number]"
+        getPlayingAnimates: {
+          "!doc":
+            "获得当前正在播放的所有(指定)动画的id列表<br/>name: 动画名;不填代表返回全部正在播放的动画<br/>返回值: 一个数组,每一项为一个正在播放的动画;可用core.stopAnimate停止播放。",
+          "!type": "fn(name?: string) -> [number]",
         },
-        "getBlockCls": {
-          "!doc": "判定某个点的图块类型<br/>例如:if(core.getBlockCls(x1, y1) != 'enemys' && core.getBlockCls(x2, y2) != 'enemy48') core.openDoor(x3, y3); // 另一个简单的机关门事件,打败或炸掉这一对不同身高的敌人就开门<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 隐藏点是否不返回null,true表示不返回null<br/>返回值:图块类型,即“地形、四帧动画、矮敌人、高敌人、道具、矮npc、高npc、自动元件、额外地形”之一",
-          "!type": "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> string"
+        getBlockCls: {
+          "!doc":
+            "判定某个点的图块类型<br/>例如:if(core.getBlockCls(x1, y1) != 'enemys' && core.getBlockCls(x2, y2) != 'enemy48') core.openDoor(x3, y3); // 另一个简单的机关门事件,打败或炸掉这一对不同身高的敌人就开门<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 隐藏点是否不返回null,true表示不返回null<br/>返回值:图块类型,即“地形、四帧动画、矮敌人、高敌人、道具、矮npc、高npc、自动元件、额外地形”之一",
+          "!type":
+            "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> string",
         },
-        "drawMap": {
-          "!doc": "地图重绘<br/>例如:core.drawMap(); // 重绘当前地图,常用于更改贴图或改变自动元件后的刷新<br/>floorId: 地图id,可省略表示当前楼层<br/>callback: 重绘完毕后的回调函数,可选",
-          "!type": "fn(floorId?: string)"
+        drawMap: {
+          "!doc":
+            "地图重绘<br/>例如:core.drawMap(); // 重绘当前地图,常用于更改贴图或改变自动元件后的刷新<br/>floorId: 地图id,可省略表示当前楼层<br/>callback: 重绘完毕后的回调函数,可选",
+          "!type": "fn(floorId?: string)",
         },
-        "nearStair": {
+        nearStair: {
           "!doc": "当前位置是否在楼梯边;在楼传平面塔模式下对箭头也有效",
-          "!type": "fn() -> bool"
+          "!type": "fn() -> bool",
         },
-        "turnBlock": {
+        turnBlock: {
           "!doc": "事件转向",
-          "!type": "fn(direction?: string, x?: number, y?: number, floorId?: string)"
+          "!type":
+            "fn(direction?: string, x?: number, y?: number, floorId?: string)",
         },
-        "getMapArray": {
-          "!doc": "生成事件层矩阵<br/>例如:core.getMapArray('MT0'); // 生成主塔0层的事件层矩阵,隐藏的图块视为0<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 可选,true表示隐藏的图块也会被表示出来<br/>返回值:事件层矩阵,注意对其阵元的访问是[y][x]",
-          "!type": "fn(floorId?: string, noCache?: bool) -> [[number]]"
+        getMapArray: {
+          "!doc":
+            "生成事件层矩阵<br/>例如:core.getMapArray('MT0'); // 生成主塔0层的事件层矩阵,隐藏的图块视为0<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 可选,true表示隐藏的图块也会被表示出来<br/>返回值:事件层矩阵,注意对其阵元的访问是[y][x]",
+          "!type": "fn(floorId?: string, noCache?: bool) -> [[number]]",
         },
-        "getMapNumber": {
+        getMapNumber: {
           "!doc": "获得事件层某个点的数字",
-          "!type": "fn(x: number, y: number, floorId?: string, noCache?: bool) -> number"
+          "!type":
+            "fn(x: number, y: number, floorId?: string, noCache?: bool) -> number",
         },
-        "jumpBlock": {
-          "!doc": "跳跃图块;从V2.7开始不再有音效<br/>例如:core.jumpBlock(0, 0, 0, 0); // 令地图左上角的图块原地跳跃半秒,再花半秒淡出<br/>sx: 起点的横坐标<br/>sy: 起点的纵坐标<br/>ex: 终点的横坐标<br/>ey: 终点的纵坐标<br/>time: 单步和淡出用时,单位为毫秒。不填视为半秒<br/>keep: 是否不淡出,true表示不淡出<br/>callback: 落地或淡出后的回调函数,可选",
-          "!type": "fn(sx: number, sy: number, ex: number, ey: number, time?: number, keep?: bool, callback?: fn())"
+        jumpBlock: {
+          "!doc":
+            "跳跃图块;从V2.7开始不再有音效<br/>例如:core.jumpBlock(0, 0, 0, 0); // 令地图左上角的图块原地跳跃半秒,再花半秒淡出<br/>sx: 起点的横坐标<br/>sy: 起点的纵坐标<br/>ex: 终点的横坐标<br/>ey: 终点的纵坐标<br/>time: 单步和淡出用时,单位为毫秒。不填视为半秒<br/>keep: 是否不淡出,true表示不淡出<br/>callback: 落地或淡出后的回调函数,可选",
+          "!type":
+            "fn(sx: number, sy: number, ex: number, ey: number, time?: number, keep?: bool, callback?: fn())",
         },
-        "replaceBlock": {
-          "!doc": "批量替换图块<br/>例如:core.replaceBlock(21, 22, core.floorIds); // 把游戏中地上当前所有的黄钥匙都变成蓝钥匙<br/>fromNumber: 旧图块的数字<br/>toNumber: 新图块的数字<br/>floorId: 地图id或其数组,不填视为当前地图",
-          "!type": "fn(fromNumber: number, toNumber: number, floorId?: string|[string])"
+        replaceBlock: {
+          "!doc":
+            "批量替换图块<br/>例如:core.replaceBlock(21, 22, core.floorIds); // 把游戏中地上当前所有的黄钥匙都变成蓝钥匙<br/>fromNumber: 旧图块的数字<br/>toNumber: 新图块的数字<br/>floorId: 地图id或其数组,不填视为当前地图",
+          "!type":
+            "fn(fromNumber: number, toNumber: number, floorId?: string|[string])",
         },
-        "drawBlock": {
+        drawBlock: {
           "!doc": "绘制一个图块",
-          "!type": "fn(block?: block, animate?: number)"
+          "!type": "fn(block?: block, animate?: number)",
         },
-        "resetMap": {
+        resetMap: {
           "!doc": "重置地图",
-          "!type": "fn(floorId?: string|[string])"
+          "!type": "fn(floorId?: string|[string])",
         },
-        "animateSetBlock": {
+        animateSetBlock: {
           "!doc": "动画形式转变某点图块",
-          "!type": "fn(number: number|string, x: number, y: number, floorId?: string, time?: number, callback?: fn())"
+          "!type":
+            "fn(number: number|string, x: number, y: number, floorId?: string, time?: number, callback?: fn())",
         },
-        "animateSetBlocks": {
+        animateSetBlocks: {
           "!doc": "动画形式同时转变若干点图块",
-          "!type": "fn(number: number|string, locs: [?], floorId?: string, time?: number, callback?: fn())"
+          "!type":
+            "fn(number: number|string, locs: [?], floorId?: string, time?: number, callback?: fn())",
         },
-        "compressMap": {
+        compressMap: {
           "!doc": "压缩地图",
-          "!type": "fn(mapArr: [[number]], floorId?: string) -> [[number]]"
+          "!type": "fn(mapArr: [[number]], floorId?: string) -> [[number]]",
         },
-        "enemyExists": {
+        enemyExists: {
           "!doc": "某个点是否存在(指定的)怪物",
-          "!type": "fn(x: number, y: number, id?: string, floorId?: string) -> bool"
+          "!type":
+            "fn(x: number, y: number, id?: string, floorId?: string) -> bool",
         },
-        "npcExists": {
+        npcExists: {
           "!doc": "某个点是否存在NPC",
-          "!type": "fn(x: number, y: number, floorId?: string) -> bool"
+          "!type": "fn(x: number, y: number, floorId?: string) -> bool",
         },
-        "getBlockByNumber": {
+        getBlockByNumber: {
           "!doc": "根据数字获得图块",
-          "!type": "fn(number: number) -> block"
+          "!type": "fn(number: number) -> block",
         },
-        "removeBlock": {
-          "!doc": "删除一个图块,对应于「隐藏事件」并同时删除<br/>例如:core.removeBlock(0, 0); // 尝试删除地图左上角的图块<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
-          "!type": "fn(x: number, y: number, floorId?: string)"
+        removeBlock: {
+          "!doc":
+            "删除一个图块,对应于「隐藏事件」并同时删除<br/>例如:core.removeBlock(0, 0); // 尝试删除地图左上角的图块<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
+          "!type": "fn(x: number, y: number, floorId?: string)",
         },
-        "hideBlock": {
-          "!doc": "隐藏一个图块,对应于「隐藏事件」且不删除<br/>例如:core.hideBlock(0, 0); // 隐藏地图左上角的图块<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
-          "!type": "fn(x: number, y: number, floorId?: string)"
+        hideBlock: {
+          "!doc":
+            "隐藏一个图块,对应于「隐藏事件」且不删除<br/>例如:core.hideBlock(0, 0); // 隐藏地图左上角的图块<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
+          "!type": "fn(x: number, y: number, floorId?: string)",
         },
-        "removeBlockByIndex": {
+        removeBlockByIndex: {
           "!doc": "根据block的索引删除该块",
-          "!type": "fn(index: number, floorId?: string)"
+          "!type": "fn(index: number, floorId?: string)",
         },
-        "stairExists": {
+        stairExists: {
           "!doc": "某个点是否存在楼梯",
-          "!type": "fn(x: number, y: number, floorId?: string) -> bool"
+          "!type": "fn(x: number, y: number, floorId?: string) -> bool",
         },
-        "isMapBlockDisabled": {
+        isMapBlockDisabled: {
           "!doc": "某个点图块是否被强制启用或禁用",
-          "!type": "fn(floorId?: string, x?: number, y?: number, flags?: ?) -> bool"
+          "!type":
+            "fn(floorId?: string, x?: number, y?: number, flags?: ?) -> bool",
         },
-        "setMapBlockDisabled": {
+        setMapBlockDisabled: {
           "!doc": "设置某个点图块的强制启用或禁用状态",
-          "!type": "fn(floorId?: string, x?: number, y?: number, disabled?: bool)"
+          "!type":
+            "fn(floorId?: string, x?: number, y?: number, disabled?: bool)",
         },
-        "setBlockOpacity": {
+        setBlockOpacity: {
           "!doc": "设置某个点图块的不透明度",
-          "!type": "fn(opacity?: number, x?: number, y?: number, floorId?: string)"
+          "!type":
+            "fn(opacity?: number, x?: number, y?: number, floorId?: string)",
         },
-        "setBlockFilter": {
+        setBlockFilter: {
           "!doc": "设置某个点图块的特效",
-          "!type": "fn(filter?: ?, x?: number, y?: number, floorId?: string)"
+          "!type": "fn(filter?: ?, x?: number, y?: number, floorId?: string)",
         },
-        "decompressMap": {
+        decompressMap: {
           "!doc": "解压缩地图",
-          "!type": "fn(mapArr: [[number]], floorId?: string) -> [[number]]"
+          "!type": "fn(mapArr: [[number]], floorId?: string) -> [[number]]",
         },
-        "automaticRoute": {
-          "!doc": "自动寻路<br/>例如:core.automaticRoute(0, 0); // 自动寻路到地图左上角<br/>destX: 目标点的横坐标<br/>destY: 目标点的纵坐标<br/>返回值:每步走完后主角的loc属性组成的一维数组",
-          "!type": "fn(destX: number, destY: number) -> [{x: number, y: number, direction: string}]"
+        automaticRoute: {
+          "!doc":
+            "自动寻路<br/>例如:core.automaticRoute(0, 0); // 自动寻路到地图左上角<br/>destX: 目标点的横坐标<br/>destY: 目标点的纵坐标<br/>返回值:每步走完后主角的loc属性组成的一维数组",
+          "!type":
+            "fn(destX: number, destY: number) -> [{x: number, y: number, direction: string}]",
         },
-        "resizeMap": {
+        resizeMap: {
           "!doc": "更改地图画布的尺寸",
-          "!type": "fn(floorId?: string)"
+          "!type": "fn(floorId?: string)",
         },
-        "getFgNumber": {
-          "!doc": "判定某点的前景层的数字<br/>例如:core.getFgNumber(); // 判断主角脚下的前景层图块的数字<br/>x: 横坐标,不填为勇士坐标<br/>y: 纵坐标,不填为勇士坐标floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存而强制重算",
-          "!type": "fn(x: number, y: number, floorId?: string, noCache?: bool) -> number"
+        getFgNumber: {
+          "!doc":
+            "判定某点的前景层的数字<br/>例如:core.getFgNumber(); // 判断主角脚下的前景层图块的数字<br/>x: 横坐标,不填为勇士坐标<br/>y: 纵坐标,不填为勇士坐标floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存而强制重算",
+          "!type":
+            "fn(x: number, y: number, floorId?: string, noCache?: bool) -> number",
         },
-        "moveBlock": {
-          "!doc": "移动图块<br/>例如:core.moveBlock(0, 0, ['down']); // 令地图左上角的图块下移一格<br/>x: 起点的横坐标<br/>y: 起点的纵坐标<br/>steps: 步伐数组<br/>time: 单步和淡出用时,单位为毫秒。不填视为半秒<br/>keep: 是否不淡出,true表示不淡出<br/>callback: 移动或淡出后的回调函数,可选",
-          "!type": "fn(x: number, y: number, steps: [string], time?: number, keep?: bool, callback?: fn())"
+        moveBlock: {
+          "!doc":
+            "移动图块<br/>例如:core.moveBlock(0, 0, ['down']); // 令地图左上角的图块下移一格<br/>x: 起点的横坐标<br/>y: 起点的纵坐标<br/>steps: 步伐数组<br/>time: 单步和淡出用时,单位为毫秒。不填视为半秒<br/>keep: 是否不淡出,true表示不淡出<br/>callback: 移动或淡出后的回调函数,可选",
+          "!type":
+            "fn(x: number, y: number, steps: [string], time?: number, keep?: bool, callback?: fn())",
         },
-        "getBgNumber": {
-          "!doc": "判定某点的背景层的数字<br/>例如:core.getBgNumber(); // 判断主角脚下的背景层图块的数字<br/>x: 横坐标,不填为勇士坐标<br/>y: 纵坐标,不填为勇士坐标<br/>floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存而强制重算",
-          "!type": "fn(x?: number, y?: number, floorId?: string, noCache?: bool) -> number"
+        getBgNumber: {
+          "!doc":
+            "判定某点的背景层的数字<br/>例如:core.getBgNumber(); // 判断主角脚下的背景层图块的数字<br/>x: 横坐标,不填为勇士坐标<br/>y: 纵坐标,不填为勇士坐标<br/>floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存而强制重算",
+          "!type":
+            "fn(x?: number, y?: number, floorId?: string, noCache?: bool) -> number",
         },
-        "getIdOfThis": {
+        getIdOfThis: {
           "!doc": "获得当前事件点的ID",
-          "!type": "fn(id?: string) -> string"
+          "!type": "fn(id?: string) -> string",
         },
-        "searchBlock": {
-          "!doc": "搜索图块, 支持通配符和正则表达式<br/>例如:core.searchBlock('*Door'); // 搜索当前地图的所有门<br/>id: 图块id,支持星号表示任意多个(0个起)字符<br/>floorId: 地图id或数组,不填视为当前地图<br/>showDisable: 隐藏点是否计入,true表示计入<br/>返回值:一个详尽的数组,一般只用到其长度",
-          "!type": "fn(id: string, floorId?: string|[string], showDisable?: bool) -> [{floorId: string, index: number, x: number, y: number, block: block}]"
+        searchBlock: {
+          "!doc":
+            "搜索图块, 支持通配符和正则表达式<br/>例如:core.searchBlock('*Door'); // 搜索当前地图的所有门<br/>id: 图块id,支持星号表示任意多个(0个起)字符<br/>floorId: 地图id或数组,不填视为当前地图<br/>showDisable: 隐藏点是否计入,true表示计入<br/>返回值:一个详尽的数组,一般只用到其长度",
+          "!type":
+            "fn(id: string, floorId?: string|[string], showDisable?: bool) -> [{floorId: string, index: number, x: number, y: number, block: block}]",
         },
-        "searchBlockWithFilter": {
-          "!doc": "根据给定的筛选函数搜索全部满足条件的图块<br/>例如:core.searchBlockWithFilter(function (block) { return block.event.id.endsWith('Door'); }); // 搜索当前地图的所有门<br/>blockFilter: 筛选函数,可接受block输入,应当返回一个boolean值<br/>floorId: 地图id或数组,不填视为当前地图<br/>showDisable: 隐藏点是否计入,true表示计入<br/>返回值:一个详尽的数组",
-          "!type": "fn(blockFilter: fn(block: block) -> bool, floorId?: string|[string], showDisable?: bool): [{floorId: string, index: number, x: number, y: number, block: block}]"
+        searchBlockWithFilter: {
+          "!doc":
+            "根据给定的筛选函数搜索全部满足条件的图块<br/>例如:core.searchBlockWithFilter(function (block) { return block.event.id.endsWith('Door'); }); // 搜索当前地图的所有门<br/>blockFilter: 筛选函数,可接受block输入,应当返回一个boolean值<br/>floorId: 地图id或数组,不填视为当前地图<br/>showDisable: 隐藏点是否计入,true表示计入<br/>返回值:一个详尽的数组",
+          "!type":
+            "fn(blockFilter: fn(block: block) -> bool, floorId?: string|[string], showDisable?: bool): [{floorId: string, index: number, x: number, y: number, block: block}]",
         },
-        "hideBgFgMap": {
+        hideBgFgMap: {
           "!doc": "隐藏前景/背景地图",
-          "!type": "fn(name?: string, loc?: [number]|[[number]], floorId?: string, callback?: fn())"
+          "!type":
+            "fn(name?: string, loc?: [number]|[[number]], floorId?: string, callback?: fn())",
         },
-        "getBlockInfo": {
-          "!doc": "获得某个图块或素材的信息,包括ID,cls,图片,坐标,faceIds等等",
-          "!type": "fn(block?: number|string|block) -> blockInfo"
+        getBlockInfo: {
+          "!doc":
+            "获得某个图块或素材的信息,包括ID,cls,图片,坐标,faceIds等等",
+          "!type": "fn(block?: number|string|block) -> blockInfo",
         },
-        "getFaceDownId": {
-          "!doc": "获得某个图块对应行走图朝向向下的那一项的id;如果不存在行走图绑定则返回自身id。",
-          "!type": "fn(block?: string|number|block) -> string"
+        getFaceDownId: {
+          "!doc":
+            "获得某个图块对应行走图朝向向下的那一项的id;如果不存在行走图绑定则返回自身id。",
+          "!type": "fn(block?: string|number|block) -> string",
         },
-        "canMoveDirectlyArray": {
+        canMoveDirectlyArray: {
           "!doc": "获得某些点可否通行的信息",
-          "!type": "fn(locs?: [[number]])"
+          "!type": "fn(locs?: [[number]])",
         },
-        "hideFloorImage": {
+        hideFloorImage: {
           "!doc": "隐藏一个楼层贴图",
-          "!type": "fn(loc?: [number]|[[number]], floorId?: string, callback?: fn())"
+          "!type":
+            "fn(loc?: [number]|[[number]], floorId?: string, callback?: fn())",
         },
-        "extractBlocks": {
+        extractBlocks: {
           "!doc": "根据需求解析出blocks",
-          "!type": "fn(map?: ?)"
+          "!type": "fn(map?: ?)",
         },
-        "extractBlocksForUI": {
+        extractBlocksForUI: {
           "!doc": "根据需求为UI解析出blocks",
-          "!type": "fn(map?: ?, flags?: ?)"
+          "!type": "fn(map?: ?, flags?: ?)",
         },
-        "getBlockId": {
-          "!doc": "判定某个点的图块id<br/>例如:if(core.getBlockId(x1, y1) != 'greenSlime' && core.getBlockId(x2, y2) != 'redSlime') core.openDoor(x3, y3); // 一个简单的机关门事件,打败或炸掉这一对绿头怪和红头怪就开门<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 隐藏点是否不返回null,true表示不返回null<br/>返回值:图块id,该点无图块则返回null",
-          "!type": "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> string"
+        getBlockId: {
+          "!doc":
+            "判定某个点的图块id<br/>例如:if(core.getBlockId(x1, y1) != 'greenSlime' && core.getBlockId(x2, y2) != 'redSlime') core.openDoor(x3, y3); // 一个简单的机关门事件,打败或炸掉这一对绿头怪和红头怪就开门<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 隐藏点是否不返回null,true表示不返回null<br/>返回值:图块id,该点无图块则返回null",
+          "!type":
+            "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> string",
         },
-        "getBlockNumber": {
-          "!doc": "判定某个点的图块数字<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 隐藏点是否不返回null,true表示不返回null<br/>返回值:图块数字,该点无图块则返回null",
-          "!type": "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> number"
+        getBlockNumber: {
+          "!doc":
+            "判定某个点的图块数字<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图<br/>showDisable: 隐藏点是否不返回null,true表示不返回null<br/>返回值:图块数字,该点无图块则返回null",
+          "!type":
+            "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> number",
         },
-        "getBlockOpacity": {
+        getBlockOpacity: {
           "!doc": "获得某个点图块的不透明度",
-          "!type": "fn(x?: number, y?: number, floorId?: string, showDisable?: bool) -> number"
+          "!type":
+            "fn(x?: number, y?: number, floorId?: string, showDisable?: bool) -> number",
         },
-        "getBlockFilter": {
+        getBlockFilter: {
           "!doc": "获得某个点图块的特效",
-          "!type": "fn(x?: number, y?: number, floorId?: string, showDisable?: bool) -> ?"
+          "!type":
+            "fn(x?: number, y?: number, floorId?: string, showDisable?: bool) -> ?",
         },
-        "loadFloor": {
+        loadFloor: {
           "!doc": "从文件或存档中加载某个楼层",
-          "!type": "fn(floorId?: string, map?: ?)"
+          "!type": "fn(floorId?: string, map?: ?)",
         },
-        "generateMovableArray": {
-          "!doc": "可通行性判定<br/>例如:core.generateMovableArray(); // 判断当前地图主角从各点能向何方向移动<br/>floorId: 地图id,不填视为当前地图<br/>返回值:从各点可移动方向的三维数组",
-          "!type": "fn(floorId?: string) -> [[[string]]]"
+        generateMovableArray: {
+          "!doc":
+            "可通行性判定<br/>例如:core.generateMovableArray(); // 判断当前地图主角从各点能向何方向移动<br/>floorId: 地图id,不填视为当前地图<br/>返回值:从各点可移动方向的三维数组",
+          "!type": "fn(floorId?: string) -> [[[string]]]",
         },
-        "terrainExists": {
+        terrainExists: {
           "!doc": "某个点是否存在(指定的)地形",
-          "!type": "fn(x: number, y: number, id?: string, floorId?: string) -> bool"
+          "!type":
+            "fn(x: number, y: number, id?: string, floorId?: string) -> bool",
         },
-        "getBlockById": {
+        getBlockById: {
           "!doc": "根据ID获得图块",
-          "!type": "fn(id: string) -> block"
+          "!type": "fn(id: string) -> block",
         },
-        "drawBg": {
-          "!doc": "绘制背景层(含贴图,其与背景层矩阵的绘制顺序可通过复写此函数来改变)<br/>例如:core.drawBg(); // 绘制当前地图的背景层<br/>floorId: 地图id,不填视为当前地图<br/>ctx: 某画布的ctx,用于绘制缩略图,一般不需要",
-          "!type": "fn(floorId?: string, ctx?: CanvasRenderingContext2D)"
+        drawBg: {
+          "!doc":
+            "绘制背景层(含贴图,其与背景层矩阵的绘制顺序可通过复写此函数来改变)<br/>例如:core.drawBg(); // 绘制当前地图的背景层<br/>floorId: 地图id,不填视为当前地图<br/>ctx: 某画布的ctx,用于绘制缩略图,一般不需要",
+          "!type": "fn(floorId?: string, ctx?: CanvasRenderingContext2D)",
         },
-        "showBlock": {
-          "!doc": "显示(隐藏或显示的)图块,此函数将被“显示事件”指令和勾选了“不消失”的“移动/跳跃事件”指令(如阻击怪)的终点调用<br/>例如:core.showBlock(0, 0); // 显示地图左上角的图块<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
-          "!type": "fn(x: number, y: number, floorId?: string)"
+        showBlock: {
+          "!doc":
+            "显示(隐藏或显示的)图块,此函数将被“显示事件”指令和勾选了“不消失”的“移动/跳跃事件”指令(如阻击怪)的终点调用<br/>例如:core.showBlock(0, 0); // 显示地图左上角的图块<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
+          "!type": "fn(x: number, y: number, floorId?: string)",
         },
-        "getMapBlocksObj": {
+        getMapBlocksObj: {
           "!doc": "以x,y的形式返回每个点的事件",
-          "!type": "fn(floorId?: string, noCache?: bool)"
+          "!type": "fn(floorId?: string, noCache?: bool)",
         },
-        "removeGlobalAnimate": {
+        removeGlobalAnimate: {
           "!doc": "删除一个或所有全局动画",
-          "!type": "fn(x?: number, y?: number, name?: string)"
+          "!type": "fn(x?: number, y?: number, name?: string)",
         },
-        "drawEvents": {
-          "!doc": "绘制事件层<br/>例如:core.drawEvents(); // 绘制当前地图的事件层<br/>floorId: 地图id,不填视为当前地图<br/>blocks: 一般不需要<br/>ctx: 某画布的ctx,用于绘制缩略图,一般不需要",
-          "!type": "fn(floorId?: string, blocks?: [block], ctx?: CanvasRenderingContext2D)"
+        drawEvents: {
+          "!doc":
+            "绘制事件层<br/>例如:core.drawEvents(); // 绘制当前地图的事件层<br/>floorId: 地图id,不填视为当前地图<br/>blocks: 一般不需要<br/>ctx: 某画布的ctx,用于绘制缩略图,一般不需要",
+          "!type":
+            "fn(floorId?: string, blocks?: [block], ctx?: CanvasRenderingContext2D)",
         },
-        "canMoveDirectly": {
-          "!doc": "能否瞬移到某点,并求出节约的步数。<br/>例如:core.canMoveDirectly(0, 0); // 能否瞬移到地图左上角<br/>destX: 目标点的横坐标<br/>destY: 目标点的纵坐标<br/>返回值:正数表示节约的步数,-1表示不可瞬移",
-          "!type": "fn(destX: number, destY: number) -> number"
+        canMoveDirectly: {
+          "!doc":
+            "能否瞬移到某点,并求出节约的步数。<br/>例如:core.canMoveDirectly(0, 0); // 能否瞬移到地图左上角<br/>destX: 目标点的横坐标<br/>destY: 目标点的纵坐标<br/>返回值:正数表示节约的步数,-1表示不可瞬移",
+          "!type": "fn(destX: number, destY: number) -> number",
         },
-        "saveMap": {
+        saveMap: {
           "!doc": "将当前地图重新变成数字,以便于存档",
-          "!type": "fn(floorId?: string)"
+          "!type": "fn(floorId?: string)",
         },
-        "drawBoxAnimate": {
+        drawBoxAnimate: {
           "!doc": "绘制UI层的box动画",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setBgFgBlock": {
-          "!doc": "转变图层块<br/>例如:core.setBgFgBlock('bg', 167, 6, 6); // 把当前地图背景层的中心块改为滑冰<br/>name: 背景还是前景<br/>number: 新图层块的数字(也支持纯数字字符串如'1')或id<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
-          "!type": "fn(name: string, number: number|string, x: number, y: number, floorId?: string)"
+        setBgFgBlock: {
+          "!doc":
+            "转变图层块<br/>例如:core.setBgFgBlock('bg', 167, 6, 6); // 把当前地图背景层的中心块改为滑冰<br/>name: 背景还是前景<br/>number: 新图层块的数字(也支持纯数字字符串如'1')或id<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
+          "!type":
+            "fn(name: string, number: number|string, x: number, y: number, floorId?: string)",
         },
-        "drawFg": {
-          "!doc": "绘制前景层(含贴图,其与前景层矩阵的绘制顺序可通过复写此函数来改变)<br/>例如:core.drawFg(); // 绘制当前地图的前景层<br/>floorId: 地图id,不填视为当前地图<br/>ctx: 某画布的ctx,用于绘制缩略图,一般不需要",
-          "!type": "fn(floorId?: string, ctx?: CanvasRenderingContext2D)"
+        drawFg: {
+          "!doc":
+            "绘制前景层(含贴图,其与前景层矩阵的绘制顺序可通过复写此函数来改变)<br/>例如:core.drawFg(); // 绘制当前地图的前景层<br/>floorId: 地图id,不填视为当前地图<br/>ctx: 某画布的ctx,用于绘制缩略图,一般不需要",
+          "!type": "fn(floorId?: string, ctx?: CanvasRenderingContext2D)",
         },
-        "getBlock": {
+        getBlock: {
           "!doc": "获得某个点的block",
-          "!type": "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> block"
+          "!type":
+            "fn(x: number, y: number, floorId?: string, showDisable?: bool) -> block",
         },
-        "initBlock": {
+        initBlock: {
           "!doc": "初始化一个图块",
-          "!type": "fn(x: number, y: number, id: string|number, addInfo?: bool, eventFloor?: ?) -> block"
+          "!type":
+            "fn(x: number, y: number, id: string|number, addInfo?: bool, eventFloor?: ?) -> block",
         },
-        "addGlobalAnimate": {
+        addGlobalAnimate: {
           "!doc": "添加一个全局动画",
-          "!type": "fn(block?: block)"
+          "!type": "fn(block?: block)",
         },
-        "animateBlock": {
+        animateBlock: {
           "!doc": "显示/隐藏某个块时的动画效果",
-          "!type": "fn(loc?: [number]|[[number]], type?: string|number, time?: number, callback?: fn())"
+          "!type":
+            "fn(loc?: [number]|[[number]], type?: string|number, time?: number, callback?: fn())",
         },
-        "loadMap": {
+        loadMap: {
           "!doc": "将存档中的地图信息重新读取出来",
-          "!type": "fn(data?: ?, floorId?: string, flags?: ?)"
+          "!type": "fn(data?: ?, floorId?: string, flags?: ?)",
         },
-        "setBlock": {
-          "!doc": "转变图块<br/>例如:core.setBlock(1, 0, 0); // 把地图左上角变成黄墙<br/>number: 新图块的数字(也支持纯数字字符串如'1')或id<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
-          "!type": "fn(number: number|string, x: number, y: number, floorId?: string)"
+        setBlock: {
+          "!doc":
+            "转变图块<br/>例如:core.setBlock(1, 0, 0); // 把地图左上角变成黄墙<br/>number: 新图块的数字(也支持纯数字字符串如'1')或id<br/>x: 横坐标<br/>y: 纵坐标<br/>floorId: 地图id,不填视为当前地图",
+          "!type":
+            "fn(number: number|string, x: number, y: number, floorId?: string)",
         },
-        "getFgMapArray": {
-          "!doc": "生成前景层矩阵<br/>例如:core.getFgMapArray('MT0'); // 生成主塔0层的前景层矩阵,使用缓存<br/>floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存<br/>返回值:前景层矩阵,注意对其阵元的访问是[y][x]",
-          "!type": "fn(floorId?: string, noCache?: bool) -> [[number]]"
+        getFgMapArray: {
+          "!doc":
+            "生成前景层矩阵<br/>例如:core.getFgMapArray('MT0'); // 生成主塔0层的前景层矩阵,使用缓存<br/>floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存<br/>返回值:前景层矩阵,注意对其阵元的访问是[y][x]",
+          "!type": "fn(floorId?: string, noCache?: bool) -> [[number]]",
         },
-        "getBgMapArray": {
-          "!doc": "生成背景层矩阵<br/>例如:core.getBgMapArray('MT0'); // 生成主塔0层的背景层矩阵,使用缓存<br/>floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存<br/>返回值:背景层矩阵,注意对其阵元的访问是[y][x]",
-          "!type": "fn(floorId?: string, noCache?: bool) -> [[number]]"
+        getBgMapArray: {
+          "!doc":
+            "生成背景层矩阵<br/>例如:core.getBgMapArray('MT0'); // 生成主塔0层的背景层矩阵,使用缓存<br/>floorId: 地图id,不填视为当前地图<br/>noCache: 可选,true表示不使用缓存<br/>返回值:背景层矩阵,注意对其阵元的访问是[y][x]",
+          "!type": "fn(floorId?: string, noCache?: bool) -> [[number]]",
         },
-        "canMoveHero": {
-          "!doc": "单点单朝向的可通行性判定;受各图层cannotInOut、起点cannotMove和canGoDeadZone影响,不受canPass和noPass影响<br/>x: 起点横坐标,不填视为主角当前的<br/>y: 起点纵坐标,不填视为主角当前的<br/>direction: 移动的方向,不填视为主角面对的方向<br/>floorId: 地图id,不填视为当前地图",
-          "!type": "fn(x?: number, y?: number, direction?: string, floorId?: string) -> bool"
+        canMoveHero: {
+          "!doc":
+            "单点单朝向的可通行性判定;受各图层cannotInOut、起点cannotMove和canGoDeadZone影响,不受canPass和noPass影响<br/>x: 起点横坐标,不填视为主角当前的<br/>y: 起点纵坐标,不填视为主角当前的<br/>direction: 移动的方向,不填视为主角面对的方向<br/>floorId: 地图id,不填视为当前地图",
+          "!type":
+            "fn(x?: number, y?: number, direction?: string, floorId?: string) -> bool",
         },
-        "drawThumbnail": {
-          "!doc": "绘制缩略图<br/>例如:core.drawThumbnail(); // 绘制当前地图的缩略图<br/>floorId: 地图id,不填视为当前地图<br/>blocks: 一般不需要<br/>options: 绘制信息,可选。可以增绘主角位置和朝向、采用不同于游戏中的主角行走图、增绘显伤、提供flags用于存读档,同时包含要绘制到的画布名或画布的ctx或还有其他信息,如起绘坐标、绘制大小、是否绘制全图、截取中心",
-          "!type": "fn(floorId?: string, blocks?: [block], options?: ?)"
+        drawThumbnail: {
+          "!doc":
+            "绘制缩略图<br/>例如:core.drawThumbnail(); // 绘制当前地图的缩略图<br/>floorId: 地图id,不填视为当前地图<br/>blocks: 一般不需要<br/>options: 绘制信息,可选。可以增绘主角位置和朝向、采用不同于游戏中的主角行走图、增绘显伤、提供flags用于存读档,同时包含要绘制到的画布名或画布的ctx或还有其他信息,如起绘坐标、绘制大小、是否绘制全图、截取中心",
+          "!type": "fn(floorId?: string, blocks?: [block], options?: ?)",
         },
-        "hideBlockByIndex": {
+        hideBlockByIndex: {
           "!doc": "根据图块的索引来隐藏图块",
-          "!type": "fn(index?: number, floorId?: string)"
+          "!type": "fn(index?: number, floorId?: string)",
         },
-        "getNumberById": {
-          "!doc": "根据图块id得到数字(地图矩阵中的值)<br/>例如:core.getNumberById('yellowWall'); // 1<br/>id: 图块id<br/>返回值:图块的数字,定义在project\\maps.js(请注意和project\\icons.js中的“图块索引”相区分!)",
-          "!type": "fn(id: string) -> number"
+        getNumberById: {
+          "!doc":
+            "根据图块id得到数字(地图矩阵中的值)<br/>例如:core.getNumberById('yellowWall'); // 1<br/>id: 图块id<br/>返回值:图块的数字,定义在project\\maps.js(请注意和project\\icons.js中的“图块索引”相区分!)",
+          "!type": "fn(id: string) -> number",
         },
-        "removeBlockByIndexes": {
+        removeBlockByIndexes: {
           "!doc": "一次性删除多个block",
-          "!type": "fn(indexes?: [number], floorId?: string)"
+          "!type": "fn(indexes?: [number], floorId?: string)",
         },
-        "hideBlockByIndexes": {
+        hideBlockByIndexes: {
           "!doc": "一次性隐藏多个block",
-          "!type": "fn(indexes?: [number], floorId?: string)"
+          "!type": "fn(indexes?: [number], floorId?: string)",
         },
-        "generateGroundPattern": {
+        generateGroundPattern: {
           "!doc": "生成groundPattern",
-          "!type": "fn(floorId?: string)"
+          "!type": "fn(floorId?: string)",
         },
-        "showBgFgMap": {
+        showBgFgMap: {
           "!doc": "显示前景/背景地图",
-          "!type": "fn(name?: string, loc?: [number]|[[number]], floorId?: string, callback?: fn())"
+          "!type":
+            "fn(name?: string, loc?: [number]|[[number]], floorId?: string, callback?: fn())",
         },
-        "showFloorImage": {
+        showFloorImage: {
           "!doc": "显示一个楼层贴图",
-          "!type": "fn(loc?: [number]|[[number]], floorId?: string, callback?: fn())"
-        }
+          "!type":
+            "fn(loc?: [number]|[[number]], floorId?: string, callback?: fn())",
+        },
       },
-      "ui": {
-        "!doc": "负责一切UI界面的绘制。主要包括三个部分:<br/>- 设置某个画布的属性与在某个画布上绘制的相关API<br/>- 具体的某个UI界面的绘制<br/>- 动态创建画布相关的API",
-        "resizeCanvas": {
+      ui: {
+        "!doc":
+          "负责一切UI界面的绘制。主要包括三个部分:<br/>- 设置某个画布的属性与在某个画布上绘制的相关API<br/>- 具体的某个UI界面的绘制<br/>- 动态创建画布相关的API",
+        resizeCanvas: {
           "!doc": "重新设置一个自定义画布的大小",
-          "!type": "fn(name: string, x?: number, y?: number, styleOnly?: bool, isTempCanvas?: bool)"
+          "!type":
+            "fn(name: string, x?: number, y?: number, styleOnly?: bool, isTempCanvas?: bool)",
         },
-        "deleteCanvas": {
-          "!doc": "删除一个自定义画布<br/>name: 画布名,也可以传入一个函数对所有画布进行筛选",
-          "!type": "fn(name: string|fn(name: string) -> bool)"
+        deleteCanvas: {
+          "!doc":
+            "删除一个自定义画布<br/>name: 画布名,也可以传入一个函数对所有画布进行筛选",
+          "!type": "fn(name: string|fn(name: string) -> bool)",
         },
-        "deleteAllCanvas": {
+        deleteAllCanvas: {
           "!doc": "清空所有的自定义画布",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "drawIcon": {
+        drawIcon: {
           "!doc": "在某个canvas上绘制一个图标",
-          "!type": "fn(name: string|CanvasRenderingContext2D, id: string, x: number, y: number, w?: number, h?: number, frame?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, id: string, x: number, y: number, w?: number, h?: number, frame?: number)",
         },
-        "drawFly": {
+        drawFly: {
           "!doc": "绘制楼层传送器",
-          "!type": "fn(page?: ?)"
+          "!type": "fn(page?: ?)",
         },
-        "setOpacity": {
-          "!doc": "设置某个canvas整体的透明度;此函数直接改变画布本身,对已经绘制的内容也生效<br/>如果仅想对接下来的绘制生效请使用setAlpha",
-          "!type": "fn(name: string|CanvasRenderingContext2D, opacity: number)"
+        setOpacity: {
+          "!doc":
+            "设置某个canvas整体的透明度;此函数直接改变画布本身,对已经绘制的内容也生效<br/>如果仅想对接下来的绘制生效请使用setAlpha",
+          "!type": "fn(name: string|CanvasRenderingContext2D, opacity: number)",
         },
-        "getTextContentHeight": {
+        getTextContentHeight: {
           "!doc": "获得某段文字的预计绘制高度;参数说明详见 drawTextContent",
-          "!type": "fn(content: string, config?: ?)"
+          "!type": "fn(content: string, config?: ?)",
         },
-        "drawArrow": {
+        drawArrow: {
           "!doc": "在某个canvas上绘制一个箭头",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x1: number, y1: number, x2: number, y2: number, style?: string, lineWidth?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x1: number, y1: number, x2: number, y2: number, style?: string, lineWidth?: number)",
         },
-        "strokeEllipse": {
+        strokeEllipse: {
           "!doc": "在某个canvas上绘制一个椭圆的边框",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, a: number, b: number, angle?: number, style?: string, lineWidth?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, a: number, b: number, angle?: number, style?: string, lineWidth?: number)",
         },
-        "fillCircle": {
+        fillCircle: {
           "!doc": "在某个canvas上绘制一个圆",
           "!url": "https://www.w3school.com.cn/tags/canvas_arc.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: number, style?: string)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: number, style?: string)",
         },
-        "strokeRoundRect": {
+        strokeRoundRect: {
           "!doc": "在某个canvas上绘制一个圆角矩形的边框",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, radius: number, style?: string, lineWidth?: number, angle?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, radius: number, style?: string, lineWidth?: number, angle?: number)",
         },
-        "getContextByName": {
-          "!doc": "根据画布名找到一个画布的context;支持系统画布和自定义画布。如果不存在画布返回null。<br/>也可以传画布的context自身,则返回自己。",
-          "!type": "fn(canvas: string|CanvasRenderingContext2D) -> CanvasRenderingContext2D"
+        getContextByName: {
+          "!doc":
+            "根据画布名找到一个画布的context;支持系统画布和自定义画布。如果不存在画布返回null。<br/>也可以传画布的context自身,则返回自己。",
+          "!type":
+            "fn(canvas: string|CanvasRenderingContext2D) -> CanvasRenderingContext2D",
         },
-        "drawImage": {
-          "!doc": "在一个画布上绘制图片<br/>后面的8个坐标参数与canvas的drawImage的八个参数完全相同。<br/>name: 可以是系统画布之一,也可以是任意自定义动态创建的画布名 画布名称或者画布的context<br/>image: 要绘制的图片,可以是一个全塔属性中定义的图片名(会从images中去获取;支持加':x',':y',':o'翻转),图片本身,或者一个画布。<br/>angle:旋转角度",
+        drawImage: {
+          "!doc":
+            "在一个画布上绘制图片<br/>后面的8个坐标参数与canvas的drawImage的八个参数完全相同。<br/>name: 可以是系统画布之一,也可以是任意自定义动态创建的画布名 画布名称或者画布的context<br/>image: 要绘制的图片,可以是一个全塔属性中定义的图片名(会从images中去获取;支持加':x',':y',':o'翻转),图片本身,或者一个画布。<br/>angle:旋转角度",
           "!url": "http://www.w3school.com.cn/html5/canvas_drawimage.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, image: string|image, x: number, y: number, w?: number, h?: number, x1?: number, y1?: number, w1?: number, h1?: number, angle?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, image: string|image, x: number, y: number, w?: number, h?: number, x1?: number, y1?: number, w1?: number, h1?: number, angle?: number)",
         },
-        "drawTip": {
-          "!doc": "左上角绘制一段提示<br/>text: 要提示的字符串,支持${}语法<br/>id: 要绘制的图标ID<br/>frame: 要绘制该图标的第几帧",
-          "!type": "fn(text: string, id?: string, frame?: number)"
+        drawTip: {
+          "!doc":
+            "左上角绘制一段提示<br/>text: 要提示的字符串,支持${}语法<br/>id: 要绘制的图标ID<br/>frame: 要绘制该图标的第几帧",
+          "!type": "fn(text: string, id?: string, frame?: number)",
         },
-        "drawBackground": {
+        drawBackground: {
           "!doc": "绘制一个背景图,可绘制winskin或纯色背景;支持小箭头绘制",
-          "!type": "fn(left: string, top: string, right: string, bottom: string, posInfo?: {px: number, py: number, direction: string})"
+          "!type":
+            "fn(left: string, top: string, right: string, bottom: string, posInfo?: {px: number, py: number, direction: string})",
         },
-        "fillEllipse": {
+        fillEllipse: {
           "!doc": "在某个canvas上绘制一个椭圆",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, a: number, b: number, angle?: number, style?: string)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, a: number, b: number, angle?: number, style?: string)",
         },
-        "setFillStyle": {
+        setFillStyle: {
           "!doc": "设置某个canvas的绘制属性(如颜色等)",
           "!url": "https://www.w3school.com.cn/tags/canvas_fillstyle.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, style: string)"
+          "!type": "fn(name: string|CanvasRenderingContext2D, style: string)",
         },
-        "drawText": {
+        drawText: {
           "!doc": "地图中间绘制一段文字",
-          "!type": "fn(contents: string, callback?: fn())"
+          "!type": "fn(contents: string, callback?: fn())",
         },
-        "drawConfirmBox": {
-          "!doc": "绘制一个确认框<br/>此项会打断事件流,如需不打断版本的请使用core.myconfirm()<br/>text: 要绘制的内容,支持 ${} 语法<br/>yesCallback: 点击确认后的回调<br/>noCallback: 点击取消后的回调",
-          "!type": "fn(text: string, yesCallback?: fn(), noCallback?: fn())"
+        drawConfirmBox: {
+          "!doc":
+            "绘制一个确认框<br/>此项会打断事件流,如需不打断版本的请使用core.myconfirm()<br/>text: 要绘制的内容,支持 ${} 语法<br/>yesCallback: 点击确认后的回调<br/>noCallback: 点击取消后的回调",
+          "!type": "fn(text: string, yesCallback?: fn(), noCallback?: fn())",
         },
-        "drawUIEventSelector": {
-          "!doc": "自绘一个闪烁的选择光标<br/>code: 选择光标的编号,必填<br/>background: 要绘制的光标背景,必须是一个合法的WindowSkin<br/>x, y, w, h: 绘制的坐标和长宽<br/>z: 可选,光标的的z值",
-          "!type": "fn(code: number, background: string, x: number, y: number, w: number, h: number, z?: number)"
+        drawUIEventSelector: {
+          "!doc":
+            "自绘一个闪烁的选择光标<br/>code: 选择光标的编号,必填<br/>background: 要绘制的光标背景,必须是一个合法的WindowSkin<br/>x, y, w, h: 绘制的坐标和长宽<br/>z: 可选,光标的的z值",
+          "!type":
+            "fn(code: number, background: string, x: number, y: number, w: number, h: number, z?: number)",
         },
-        "clearUIEventSelector": {
-          "!doc": "清除若干个自绘的选择光标<br/>codes: 清除的光标编号;可以是单个编号或编号数组;不填则清除所有光标",
-          "!type": "fn(codes?: number|[number])"
+        clearUIEventSelector: {
+          "!doc":
+            "清除若干个自绘的选择光标<br/>codes: 清除的光标编号;可以是单个编号或编号数组;不填则清除所有光标",
+          "!type": "fn(codes?: number|[number])",
         },
-        "fillPolygon": {
+        fillPolygon: {
           "!doc": "在某个canvas上绘制一个多边形",
-          "!type": "fn(name: string|CanvasRenderingContext2D, nodes?: [[number]], style?: string)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, nodes?: [[number]], style?: string)",
         },
-        "fillText": {
-          "!doc": "在某个画布上绘制一段文字<br/>text: 要绘制的文本<br/>style: 绘制的样式<br/>font: 绘制的字体<br/>最大宽度,超过此宽度会自动放缩",
+        fillText: {
+          "!doc":
+            "在某个画布上绘制一段文字<br/>text: 要绘制的文本<br/>style: 绘制的样式<br/>font: 绘制的字体<br/>最大宽度,超过此宽度会自动放缩",
           "!url": "https://www.w3school.com.cn/tags/canvas_filltext.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, text: string, x: number, y: number, style?: string, font?: string, maxWidth?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, text: string, x: number, y: number, style?: string, font?: string, maxWidth?: number)",
         },
-        "setTextBaseline": {
-          "!doc": "设置某个canvas的基准线<br/>baseline: 可为alphabetic, top, hanging, middle, ideographic, bottom",
+        setTextBaseline: {
+          "!doc":
+            "设置某个canvas的基准线<br/>baseline: 可为alphabetic, top, hanging, middle, ideographic, bottom",
           "!url": "https://www.w3school.com.cn/tags/canvas_textbaseline.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, baseline: string)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, baseline: string)",
         },
-        "loadCanvas": {
+        loadCanvas: {
           "!doc": "加载某个canvas状态",
-          "!type": "fn(name: string|CanvasRenderingContext2D)"
+          "!type": "fn(name: string|CanvasRenderingContext2D)",
         },
-        "splitLines": {
+        splitLines: {
           "!doc": "字符串自动换行的分割",
-          "!type": "fn(name: string|CanvasRenderingContext2D, text: string, maxWidth?: number, font?: string)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, text: string, maxWidth?: number, font?: string)",
         },
-        "setAlpha": {
-          "!doc": "设置某个canvas接下来绘制的不透明度;不会影响已经绘制的内容<br/>返回设置之前画布的不透明度<br/>如果需要修改画布本身的不透明度请使用setOpacity",
+        setAlpha: {
+          "!doc":
+            "设置某个canvas接下来绘制的不透明度;不会影响已经绘制的内容<br/>返回设置之前画布的不透明度<br/>如果需要修改画布本身的不透明度请使用setOpacity",
           "!url": "https://www.w3school.com.cn/tags/canvas_globalalpha.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, alpha: number) -> number"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, alpha: number) -> number",
         },
-        "setFilter": {
+        setFilter: {
           "!doc": "设置某个canvas接下来绘制的filter",
-          "!type": "fn(name: string|CanvasRenderingContext2D, style: string)"
+          "!type": "fn(name: string|CanvasRenderingContext2D, style: string)",
         },
-        "setLineWidth": {
+        setLineWidth: {
           "!doc": "设置某个canvas的线宽度",
           "!url": "https://www.w3school.com.cn/tags/canvas_linewidth.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, lineWidth: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, lineWidth: number)",
         },
-        "drawTextBox": {
+        drawTextBox: {
           "!doc": "绘制一个对话框",
-          "!type": "fn(content: string, showAll?: bool)"
+          "!type": "fn(content: string, showAll?: bool)",
         },
-        "relocateCanvas": {
+        relocateCanvas: {
           "!doc": "重新定位一个自定义画布",
-          "!type": "fn(name: string, x: number, y: number, useDelta: bool)"
+          "!type": "fn(name: string, x: number, y: number, useDelta: bool)",
         },
-        "rotateCanvas": {
-          "!doc": "设置一个自定义画布的旋转角度<br/>centerX, centerY: 旋转中心(以屏幕像素为基准);不填视为图片正中心。",
-          "!type": "fn(name: string, angle: number, centerX?: number, centerY?: number)"
+        rotateCanvas: {
+          "!doc":
+            "设置一个自定义画布的旋转角度<br/>centerX, centerY: 旋转中心(以屏幕像素为基准);不填视为图片正中心。",
+          "!type":
+            "fn(name: string, angle: number, centerX?: number, centerY?: number)",
         },
-        "closePanel": {
+        closePanel: {
           "!doc": "结束一切事件和绘制,关闭UI窗口,返回游戏进程",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "textImage": {
+        textImage: {
           "!doc": "文本图片化",
-          "!type": "fn(content: string, lineHeight?: number) -> image"
+          "!type": "fn(content: string, lineHeight?: number) -> image",
         },
-        "drawStatusBar": {
+        drawStatusBar: {
           "!doc": "绘制状态栏",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "setStrokeStyle": {
+        setStrokeStyle: {
           "!doc": "设置某个canvas边框属性",
           "!url": "https://www.w3school.com.cn/tags/canvas_strokestyle.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, style: string)"
+          "!type": "fn(name: string|CanvasRenderingContext2D, style: string)",
         },
-        "clearUI": {
+        clearUI: {
           "!doc": "清空UI层内容",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "drawWindowSkin": {
+        drawWindowSkin: {
           "!doc": "绘制WindowSkin",
-          "!type": "fn(background: string, ctx: string|CanvasRenderingContext2D, x: number, y: number, w: string, h: string, direction?: string, px?: number, py?: number)"
+          "!type":
+            "fn(background: string, ctx: string|CanvasRenderingContext2D, x: number, y: number, w: string, h: string, direction?: string, px?: number, py?: number)",
         },
-        "fillRect": {
-          "!doc": "绘制一个矩形。<br/>x,y: 绘制的坐标<br/>width,height: 绘制的长宽<br/>style: 绘制的样式<br/>angle: 旋转的角度,弧度制,如Math.PI/2代表90度",
+        fillRect: {
+          "!doc":
+            "绘制一个矩形。<br/>x,y: 绘制的坐标<br/>width,height: 绘制的长宽<br/>style: 绘制的样式<br/>angle: 旋转的角度,弧度制,如Math.PI/2代表90度",
           "!url": "https://www.w3school.com.cn/tags/canvas_fillrect.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, style?: string, angle?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, style?: string, angle?: number)",
         },
-        "drawScrollText": {
+        drawScrollText: {
           "!doc": "绘制滚动字幕",
-          "!type": "fn(content: string, time: number, lineHeight?: number, callback?: fn())"
+          "!type":
+            "fn(content: string, time: number, lineHeight?: number, callback?: fn())",
         },
-        "strokePolygon": {
+        strokePolygon: {
           "!doc": "在某个canvas上绘制一个多边形的边框",
-          "!type": "fn(name: string|CanvasRenderingContext2D, nodes?: [[number]], style?: string, lineWidth?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, nodes?: [[number]], style?: string, lineWidth?: number)",
         },
-        "strokeCircle": {
+        strokeCircle: {
           "!doc": "在某个canvas上绘制一个圆的边框",
           "!url": "https://www.w3school.com.cn/tags/canvas_arc.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: ?, style?: string, lineWidth?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: ?, style?: string, lineWidth?: number)",
         },
-        "drawWaiting": {
+        drawWaiting: {
           "!doc": "绘制等待界面",
-          "!type": "fn(text: string)"
+          "!type": "fn(text: string)",
         },
-        "setFont": {
+        setFont: {
           "!doc": "设置某个canvas的文字字体",
           "!url": "https://www.w3school.com.cn/tags/canvas_font.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, font: string)"
+          "!type": "fn(name: string|CanvasRenderingContext2D, font: string)",
         },
-        "drawChoices": {
+        drawChoices: {
           "!doc": "绘制一个选项界面",
-          "!type": "fn(content?: string, choices?: [?], width?: number, ctx?: string|CanvasRenderingContext2D)"
+          "!type":
+            "fn(content?: string, choices?: [?], width?: number, ctx?: string|CanvasRenderingContext2D)",
         },
-        "setFontForMaxWidth": {
+        setFontForMaxWidth: {
           "!doc": "根据最大宽度自动缩小字体",
-          "!type": "fn(name: string|CanvasRenderingContext2D, text: string, maxWidth: number, font?: ?) -> string"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, text: string, maxWidth: number, font?: ?) -> string",
         },
-        "clearMap": {
-          "!doc": "清空某个画布图层<br/>name为画布名,可以是系统画布之一,也可以是任意自定义动态创建的画布名;还可以直接传画布的context本身。<br/>如果name也可以是'all',若为all则为清空所有系统画布。",
+        clearMap: {
+          "!doc":
+            "清空某个画布图层<br/>name为画布名,可以是系统画布之一,也可以是任意自定义动态创建的画布名;还可以直接传画布的context本身。<br/>如果name也可以是'all',若为all则为清空所有系统画布。",
           "!url": "https://www.w3school.com.cn/tags/canvas_clearrect.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x?: number, y?: number, width?: number, height?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x?: number, y?: number, width?: number, height?: number)",
         },
-        "drawTextContent": {
-          "!doc": "绘制一段文字到某个画布上面<br/>ctx: 要绘制到的画布<br/>content: 要绘制的内容;转义字符不允许保留 \\t, \\b 和 \\f<br/>config: 绘制配置项,目前暂时包含如下内容(均为可选)<br/>left, top:起始点位置;maxWidth:单行最大宽度;color:默认颜色;align:左中右<br/>fontSize:字体大小;lineHeight:行高;time:打字机间隔;font:字体名<br/>返回值:绘制信息",
-          "!type": "fn(ctx: string|CanvasRenderingContext2D, content: string, config: ?)"
+        drawTextContent: {
+          "!doc":
+            "绘制一段文字到某个画布上面<br/>ctx: 要绘制到的画布<br/>content: 要绘制的内容;转义字符不允许保留 \\t, \\b 和 \\f<br/>config: 绘制配置项,目前暂时包含如下内容(均为可选)<br/>left, top:起始点位置;maxWidth:单行最大宽度;color:默认颜色;align:左中右<br/>fontSize:字体大小;lineHeight:行高;time:打字机间隔;font:字体名<br/>返回值:绘制信息",
+          "!type":
+            "fn(ctx: string|CanvasRenderingContext2D, content: string, config: ?)",
         },
-        "calWidth": {
+        calWidth: {
           "!doc": "计算某段文字的宽度",
           "!url": "https://www.w3school.com.cn/tags/canvas_measuretext.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, text: string, font?: string) -> number"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, text: string, font?: string) -> number",
         },
-        "fillArc": {
+        fillArc: {
           "!doc": "在某个canvas上绘制一个扇形",
           "!url": "https://www.w3school.com.cn/tags/canvas_arc.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: number, start: number, end: number, style?: string)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: number, start: number, end: number, style?: string)",
         },
-        "strokeArc": {
+        strokeArc: {
           "!doc": "在某个canvas上绘制一段弧",
           "!url": "https://www.w3school.com.cn/tags/canvas_arc.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: number, start: number, end: number, style?: string, lineWidth?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, r: number, start: number, end: number, style?: string, lineWidth?: number)",
         },
-        "drawLine": {
+        drawLine: {
           "!doc": "在某个canvas上绘制一条线",
           "!url": "https://www.w3school.com.cn/tags/canvas_lineto.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x1: number, y1: number, x2: number, y2: number, style?: string, lineWidth?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x1: number, y1: number, x2: number, y2: number, style?: string, lineWidth?: number)",
         },
-        "drawPagination": {
+        drawPagination: {
           "!doc": "绘制分页",
-          "!type": "fn(page?: ?, totalPage?: ?, y?: number)"
+          "!type": "fn(page?: ?, totalPage?: ?, y?: number)",
         },
-        "getToolboxItems": {
+        getToolboxItems: {
           "!doc": "获得所有应该在道具栏显示的某个类型道具",
-          "!type": "fn(cls: string) -> [string]"
+          "!type": "fn(cls: string) -> [string]",
         },
-        "strokeRect": {
-          "!doc": "绘制一个矩形的边框<br/>style: 绘制的样式<br/>lineWidth: 线宽<br/>angle: 旋转角度,弧度制,如Math.PI/2为90度",
+        strokeRect: {
+          "!doc":
+            "绘制一个矩形的边框<br/>style: 绘制的样式<br/>lineWidth: 线宽<br/>angle: 旋转角度,弧度制,如Math.PI/2为90度",
           "!url": "https://www.w3school.com.cn/tags/canvas_strokerect.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, style?: string, lineWidth?: number, angle?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, style?: string, lineWidth?: number, angle?: number)",
         },
-        "drawBook": {
+        drawBook: {
           "!doc": "绘制怪物手册",
-          "!type": "fn(index?: ?)"
+          "!type": "fn(index?: ?)",
         },
-        "fillRoundRect": {
+        fillRoundRect: {
           "!doc": "在某个canvas上绘制一个圆角矩形",
-          "!type": "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, radius: number, style?: string, angle?: number)"
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, x: number, y: number, width: number, height: number, radius: number, style?: string, angle?: number)",
         },
-        "fillBoldText": {
-          "!doc": "在某个画布上绘制一个描边文字<br/>text: 要绘制的文本<br/>style: 绘制的样式<br/>strokeStyle: 要绘制的描边颜色<br/>font: 绘制的字体<br/>maxWidth: 最大宽度,超过此宽度会自动放缩",
-          "!type": "fn(name: string|CanvasRenderingContext2D, text: string, x: number, y: number, style?: string, strokeStyle?: string, font?: string, maxWidth?: number)"
+        fillBoldText: {
+          "!doc":
+            "在某个画布上绘制一个描边文字<br/>text: 要绘制的文本<br/>style: 绘制的样式<br/>strokeStyle: 要绘制的描边颜色<br/>font: 绘制的字体<br/>maxWidth: 最大宽度,超过此宽度会自动放缩",
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, text: string, x: number, y: number, style?: string, strokeStyle?: string, font?: string, maxWidth?: number)",
         },
-        "saveCanvas": {
+        saveCanvas: {
           "!doc": "保存某个canvas状态",
-          "!type": "fn(name: string|CanvasRenderingContext2D)"
+          "!type": "fn(name: string|CanvasRenderingContext2D)",
         },
-        "createCanvas": {
-          "!doc": "动态创建一个画布。<br/>name: 要创建的画布名,如果已存在则会直接取用当前存在的。<br/>x,y: 创建的画布相对窗口左上角的像素坐标<br/>width,height: 创建的长宽。<br/>zIndex: 创建的纵向高度(关系到画布之间的覆盖),z值高的将覆盖z值低的;系统画布的z值可在个性化中查看。<br/>返回创建的画布的context,也可以通过core.dymCanvas[name]调用。",
-          "!type": "fn(name: string, x: number, y: number, width: number, height: number, zIndex: number) -> CanvasRenderingContext2D"
+        createCanvas: {
+          "!doc":
+            "动态创建一个画布。<br/>name: 要创建的画布名,如果已存在则会直接取用当前存在的。<br/>x,y: 创建的画布相对窗口左上角的像素坐标<br/>width,height: 创建的长宽。<br/>zIndex: 创建的纵向高度(关系到画布之间的覆盖),z值高的将覆盖z值低的;系统画布的z值可在个性化中查看。<br/>返回创建的画布的context,也可以通过core.dymCanvas[name]调用。",
+          "!type":
+            "fn(name: string, x: number, y: number, width: number, height: number, zIndex: number) -> CanvasRenderingContext2D",
         },
-        "setTextAlign": {
+        setTextAlign: {
           "!doc": "设置某个canvas的对齐",
           "!url": "https://www.w3school.com.cn/tags/canvas_textalign.asp",
-          "!type": "fn(name: string|CanvasRenderingContext2D, align: string)"
+          "!type": "fn(name: string|CanvasRenderingContext2D, align: string)",
         },
       },
-      "enemys": {
+      enemys: {
         "!doc": "定义了一系列和怪物相关的API函数。",
-        "getEnemys": {
-          "!doc": "获得所有怪物原始数据的一个副本。<br/>请使用core.material.enemys获得当前各项怪物属性。",
-          "!type": "fn()"
+        getEnemys: {
+          "!doc":
+            "获得所有怪物原始数据的一个副本。<br/>请使用core.material.enemys获得当前各项怪物属性。",
+          "!type": "fn()",
         },
-        "getEnemyValue": {
+        getEnemyValue: {
           "!doc": "获得某个点上怪物的某个属性值",
-          "!type": "fn(enemy?: string|enemy, name: string, x?: number, y?: number, floorId?: string)"
+          "!type":
+            "fn(enemy?: string|enemy, name: string, x?: number, y?: number, floorId?: string)",
         },
-        "getSpecials": {
+        getSpecials: {
           "!doc": "获得所有特殊属性的定义",
-          "!type": "fn() -> [[?]]"
+          "!type": "fn() -> [[?]]",
         },
-        "getSpecialColor": {
+        getSpecialColor: {
           "!doc": "获得某个怪物所有特殊属性的颜色",
-          "!type": "fn(enemy: string|enemy) -> [string]"
+          "!type": "fn(enemy: string|enemy) -> [string]",
         },
-        "getSpecialFlag": {
-          "!doc": "获得某个怪物所有特殊属性的额外标记。<br/><br/>例如,1为全图性技能,需要进行遍历全图(光环/支援等)",
-          "!type": "fn(enemy: string|enemy) -> number"
+        getSpecialFlag: {
+          "!doc":
+            "获得某个怪物所有特殊属性的额外标记。<br/><br/>例如,1为全图性技能,需要进行遍历全图(光环/支援等)",
+          "!type": "fn(enemy: string|enemy) -> number",
         },
-        "getSpecialHint": {
-          "!doc": "获得某种敌人的某种特殊属性的介绍<br/>例如:core.getSpecialHint('bat', 1) // '先攻:怪物首先攻击'<br/>enemy: 敌人id或敌人对象,用于确定属性的具体数值,否则可选<br/>special: 属性编号,可以是该敌人没有的属性<br/>返回值:属性的介绍,以属性名加中文冒号开头",
-          "!type": "fn(enemy: string|enemy, special: number) -> string"
+        getSpecialHint: {
+          "!doc":
+            "获得某种敌人的某种特殊属性的介绍<br/>例如:core.getSpecialHint('bat', 1) // '先攻:怪物首先攻击'<br/>enemy: 敌人id或敌人对象,用于确定属性的具体数值,否则可选<br/>special: 属性编号,可以是该敌人没有的属性<br/>返回值:属性的介绍,以属性名加中文冒号开头",
+          "!type": "fn(enemy: string|enemy, special: number) -> string",
         },
-        "getSpecialText": {
-          "!doc": "获得某种敌人的全部特殊属性名称<br/>例如:core.getSpecialText('greenSlime') // ['先攻', '3连击', '破甲', '反击']<br/>enemy: 敌人id或敌人对象,如core.material.enemys.greenSlime<br/>返回值:字符串数组",
-          "!type": "fn(enemy: string|enemy) -> [string]"
+        getSpecialText: {
+          "!doc":
+            "获得某种敌人的全部特殊属性名称<br/>例如:core.getSpecialText('greenSlime') // ['先攻', '3连击', '破甲', '反击']<br/>enemy: 敌人id或敌人对象,如core.material.enemys.greenSlime<br/>返回值:字符串数组",
+          "!type": "fn(enemy: string|enemy) -> [string]",
         },
-        "hasSpecial": {
-          "!doc": "判定某种特殊属性的有无<br/>例如:core.hasSpecial('greenSlime', 1) // 判定绿头怪有无先攻属性<br/>special: 敌人id或敌人对象或正整数数组或自然数<br/>test: 待检查的属性编号<br/>",
-          "!type": "fn(special: number|[number]|string|number, test: number) -> bool"
+        hasSpecial: {
+          "!doc":
+            "判定某种特殊属性的有无<br/>例如:core.hasSpecial('greenSlime', 1) // 判定绿头怪有无先攻属性<br/>special: 敌人id或敌人对象或正整数数组或自然数<br/>test: 待检查的属性编号<br/>",
+          "!type":
+            "fn(special: number|[number]|string|number, test: number) -> bool",
         },
-        "nextCriticals": {
-          "!doc": "获得某只敌人接下来的若干个临界及其减伤,算法基于useLoop开关选择回合法或二分法<br/>例如:core.nextCriticals('greenSlime', 9, 0, 0, 'MT0') // 绿头怪接下来的9个临界<br/>enemy: 敌人id或敌人对象<br/>number: 要计算的临界数量,可选,默认为1<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回:两列的二维数组,每行表示一个临界及其减伤",
-          "!type": "fn(enemy: string|enemy, number?: number, x?: number, y?: number, floorId?: string) -> [[number]]"
+        nextCriticals: {
+          "!doc":
+            "获得某只敌人接下来的若干个临界及其减伤,算法基于useLoop开关选择回合法或二分法<br/>例如:core.nextCriticals('greenSlime', 9, 0, 0, 'MT0') // 绿头怪接下来的9个临界<br/>enemy: 敌人id或敌人对象<br/>number: 要计算的临界数量,可选,默认为1<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回:两列的二维数组,每行表示一个临界及其减伤",
+          "!type":
+            "fn(enemy: string|enemy, number?: number, x?: number, y?: number, floorId?: string) -> [[number]]",
         },
-        "getDefDamage": {
-          "!doc": "计算再加若干点防御能使某只敌人对主角的总伤害降低多少<br/>例如:core.getDefDamage('greenSlime', 10, 0, 0, 'MT0') // 再加10点防御能使绿头怪的伤害降低多少<br/>enemy: 敌人id或敌人对象<br/>k: 假设主角增加的防御力,可选,默认为1<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选",
-          "!type": "fn(enemy: string|enemy, k?: number, x?: number, y?: number, floorId?: string) -> number"
+        getDefDamage: {
+          "!doc":
+            "计算再加若干点防御能使某只敌人对主角的总伤害降低多少<br/>例如:core.getDefDamage('greenSlime', 10, 0, 0, 'MT0') // 再加10点防御能使绿头怪的伤害降低多少<br/>enemy: 敌人id或敌人对象<br/>k: 假设主角增加的防御力,可选,默认为1<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选",
+          "!type":
+            "fn(enemy: string|enemy, k?: number, x?: number, y?: number, floorId?: string) -> number",
         },
-        "canBattle": {
-          "!doc": "判定主角当前能否打败某只敌人<br/>例如:core.canBattle('greenSlime',0,0,'MT0') // 能否打败主塔0层左上角的绿头怪(假设有)<br/>enemy: 敌人id或敌人对象<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:true表示可以打败,false表示无法打败",
-          "!type": "fn(enemy: string|enemy, x?: number, y?: number, floorId?: string) -> bool"
+        canBattle: {
+          "!doc":
+            "判定主角当前能否打败某只敌人<br/>例如:core.canBattle('greenSlime',0,0,'MT0') // 能否打败主塔0层左上角的绿头怪(假设有)<br/>enemy: 敌人id或敌人对象<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:true表示可以打败,false表示无法打败",
+          "!type":
+            "fn(enemy: string|enemy, x?: number, y?: number, floorId?: string) -> bool",
         },
-        "getEnemyInfo": {
-          "!doc": "获得怪物真实属性<br/>hero: 可选,此时的勇士属性<br/>此函数将会计算包括坚固、模仿、光环等若干效果,将同时被怪物手册和伤害计算调用",
-          "!type": "fn(enemy: string|enemy, hero?: ?, x?: number, y?: number, floorId?: string) -> {hp: number, atk: number, def: number, money: number, exp: number, special: [number], point: number, guards: [?]}"
+        getEnemyInfo: {
+          "!doc":
+            "获得怪物真实属性<br/>hero: 可选,此时的勇士属性<br/>此函数将会计算包括坚固、模仿、光环等若干效果,将同时被怪物手册和伤害计算调用",
+          "!type":
+            "fn(enemy: string|enemy, hero?: ?, x?: number, y?: number, floorId?: string) -> {hp: number, atk: number, def: number, money: number, exp: number, special: [number], point: number, guards: [?]}",
         },
-        "getDamageInfo": {
-          "!doc": "获得战斗伤害信息<br/>例如:core.getDamage('greenSlime',0,0,'MT0') // 绿头怪的总伤害<br/>enemy: 敌人id或敌人对象<br/>hero: 可选,此时的勇士属性<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:伤害计算信息,如果因为没有破防或无敌怪等其他原因无法战斗,则返回null",
-          "!type": "fn(enemy: string|enemy, hero?: ?, x?: number, y?: number, floorId?: string) -> {damage: number, per_damage: number, hero_per_damage: number, init_damage: number, mon_hp: number, mon_atk: number, mon_def: number, turn: number}"
+        getDamageInfo: {
+          "!doc":
+            "获得战斗伤害信息<br/>例如:core.getDamage('greenSlime',0,0,'MT0') // 绿头怪的总伤害<br/>enemy: 敌人id或敌人对象<br/>hero: 可选,此时的勇士属性<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:伤害计算信息,如果因为没有破防或无敌怪等其他原因无法战斗,则返回null",
+          "!type":
+            "fn(enemy: string|enemy, hero?: ?, x?: number, y?: number, floorId?: string) -> {damage: number, per_damage: number, hero_per_damage: number, init_damage: number, mon_hp: number, mon_atk: number, mon_def: number, turn: number}",
         },
-        "getDamage": {
-          "!doc": "获得某只敌人对主角的总伤害<br/>例如:core.getDamage('greenSlime',0,0,'MT0') // 绿头怪的总伤害<br/>enemy: 敌人id或敌人对象<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:总伤害,如果因为没有破防或无敌怪等其他原因无法战斗,则返回null",
-          "!type": "fn(enemy: string|enemy, x?: number, y?: number, floorId?: string) -> number"
+        getDamage: {
+          "!doc":
+            "获得某只敌人对主角的总伤害<br/>例如:core.getDamage('greenSlime',0,0,'MT0') // 绿头怪的总伤害<br/>enemy: 敌人id或敌人对象<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:总伤害,如果因为没有破防或无敌怪等其他原因无法战斗,则返回null",
+          "!type":
+            "fn(enemy: string|enemy, x?: number, y?: number, floorId?: string) -> number",
         },
-        "getDamageString": {
-          "!doc": "获得某只敌人的地图显伤,包括颜色<br/>例如:core.getDamageString('greenSlime', 0, 0, 'MT0') // 绿头怪的地图显伤<br/>enemy: 敌人id或敌人对象<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:damage: 表示伤害值或为'???',color: 形如'#RrGgBb'",
-          "!type": "fn(enemy: string|enemy, x?: number, y?: number, floorId?: string) -> {color: string, damage: string}"
+        getDamageString: {
+          "!doc":
+            "获得某只敌人的地图显伤,包括颜色<br/>例如:core.getDamageString('greenSlime', 0, 0, 'MT0') // 绿头怪的地图显伤<br/>enemy: 敌人id或敌人对象<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>floorId: 敌人所在的地图,可选<br/>返回值:damage: 表示伤害值或为'???',color: 形如'#RrGgBb'",
+          "!type":
+            "fn(enemy: string|enemy, x?: number, y?: number, floorId?: string) -> {color: string, damage: string}",
         },
-        "getCurrentEnemys": {
-          "!doc": "获得某张地图的敌人集合,用于手册绘制<br/>例如:core.getCurrentEnemys('MT0') // 主塔0层的敌人集合<br/>floorId: 地图id,可选<br/>返回值:敌人集合,按伤害升序排列,支持多朝向怪合并",
-          "!type": "fn(floorId?: string) -> [enemy]"
+        getCurrentEnemys: {
+          "!doc":
+            "获得某张地图的敌人集合,用于手册绘制<br/>例如:core.getCurrentEnemys('MT0') // 主塔0层的敌人集合<br/>floorId: 地图id,可选<br/>返回值:敌人集合,按伤害升序排列,支持多朝向怪合并",
+          "!type": "fn(floorId?: string) -> [enemy]",
+        },
+        hasEnemyLeft: {
+          "!doc":
+            "检查某些楼层是否还有漏打的(某种)敌人<br/>例如:core.hasEnemyLeft('greenSlime', ['sample0', 'sample1']) // 样板0层和1层是否有漏打的绿头怪<br/>enemyId: 敌人id,可选,null表示任意敌人<br/>floorId: 地图id或其数组,可选,不填为当前地图<br/>返回值:地图中是否还存在该种敌人",
+          "!type": "fn(enemyId?: string, floorId?: string|[string]) -> bool",
         },
-        "hasEnemyLeft": {
-          "!doc": "检查某些楼层是否还有漏打的(某种)敌人<br/>例如:core.hasEnemyLeft('greenSlime', ['sample0', 'sample1']) // 样板0层和1层是否有漏打的绿头怪<br/>enemyId: 敌人id,可选,null表示任意敌人<br/>floorId: 地图id或其数组,可选,不填为当前地图<br/>返回值:地图中是否还存在该种敌人",
-          "!type": "fn(enemyId?: string, floorId?: string|[string]) -> bool"
-        }
       },
-      "events": {
-        "!doc": "events.js将处理所有和事件相关的操作,主要分为五个部分:<br/>- 游戏的开始和结束<br/>- 系统事件的处理<br/>- 自定义事件的处理<br/>- 点击状态栏图标所进行的操作<br/>- 一些具体事件的执行内容",
-        "afterChangeFloor": {
+      events: {
+        "!doc":
+          "events.js将处理所有和事件相关的操作,主要分为五个部分:<br/>- 游戏的开始和结束<br/>- 系统事件的处理<br/>- 自定义事件的处理<br/>- 点击状态栏图标所进行的操作<br/>- 一些具体事件的执行内容",
+        afterChangeFloor: {
           "!doc": "转换楼层结束的事件",
-          "!type": "fn(floorId?: string)"
+          "!type": "fn(floorId?: string)",
         },
-        "popEventLoc": {
+        popEventLoc: {
           "!doc": "将当前点坐标入栈",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "afterOpenDoor": {
+        afterOpenDoor: {
           "!doc": "开一个门后触发的事件",
-          "!type": "fn(doorId?: string, x?: number, y?: number)"
+          "!type": "fn(doorId?: string, x?: number, y?: number)",
         },
-        "checkLvUp": {
+        checkLvUp: {
           "!doc": "检查升级事件",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "insertAction": {
-          "!doc": "插入一段事件;此项不可插入公共事件,请用 core.insertCommonEvent<br/>例如:core.insertAction('一段文字'); // 插入一个显示文章<br/>action: 单个事件指令,或事件指令数组<br/>x: 新的当前点横坐标,可选<br/>y: 新的当前点纵坐标,可选<br/>callback: 新的回调函数,可选<br/>addToLast: 插入的位置,true表示插入到末尾,否则插入到开头",
-          "!type": "fn(action: string|?|[?], x?: number, y?: number, callback?: fn(), addToLast?: bool)"
+        insertAction: {
+          "!doc":
+            "插入一段事件;此项不可插入公共事件,请用 core.insertCommonEvent<br/>例如:core.insertAction('一段文字'); // 插入一个显示文章<br/>action: 单个事件指令,或事件指令数组<br/>x: 新的当前点横坐标,可选<br/>y: 新的当前点纵坐标,可选<br/>callback: 新的回调函数,可选<br/>addToLast: 插入的位置,true表示插入到末尾,否则插入到开头",
+          "!type":
+            "fn(action: string|?|[?], x?: number, y?: number, callback?: fn(), addToLast?: bool)",
         },
-        "unfollow": {
+        unfollow: {
           "!doc": "取消跟随<br/>name: 取消跟随的行走图,不填则取消全部跟随者",
-          "!type": "fn(name?: string)"
+          "!type": "fn(name?: string)",
         },
-        "hasVisitedFloor": {
+        hasVisitedFloor: {
           "!doc": "是否到达过某个楼层",
-          "!type": "fn(floorId?: string) -> bool"
+          "!type": "fn(floorId?: string) -> bool",
         },
-        "startEvents": {
+        startEvents: {
           "!doc": "开始执行一系列自定义事件",
-          "!type": "fn(list?: [?], x?: number, y?: number, callback?: fn())"
+          "!type": "fn(list?: [?], x?: number, y?: number, callback?: fn())",
         },
-        "setHeroIcon": {
-          "!doc": "更改主角行走图<br/>例如:core.setHeroIcon('npc48.png', true); // 把主角从阳光变成样板0层左下角的小姐姐,但不立即刷新<br/>name: 新的行走图文件名,可以是全塔属性中映射前的中文名。映射后会被存入core.status.hero.image<br/>noDraw: true表示不立即刷新(刷新会导致大地图下视野重置到以主角为中心)",
-          "!type": "fn(name: string, noDraw?: bool)"
+        setHeroIcon: {
+          "!doc":
+            "更改主角行走图<br/>例如:core.setHeroIcon('npc48.png', true); // 把主角从阳光变成样板0层左下角的小姐姐,但不立即刷新<br/>name: 新的行走图文件名,可以是全塔属性中映射前的中文名。映射后会被存入core.status.hero.image<br/>noDraw: true表示不立即刷新(刷新会导致大地图下视野重置到以主角为中心)",
+          "!type": "fn(name: string, noDraw?: bool)",
         },
-        "changingFloor": {
+        changingFloor: {
           "!doc": "楼层转换中",
-          "!type": "fn(floorId?: string, heroLoc?: {x: number, y: number, direction: string})"
+          "!type":
+            "fn(floorId?: string, heroLoc?: {x: number, y: number, direction: string})",
         },
-        "setEvents": {
+        setEvents: {
           "!doc": "直接设置事件列表",
-          "!type": "fn(list?: [?], x?: number, y?: number, callback?: fn())"
+          "!type": "fn(list?: [?], x?: number, y?: number, callback?: fn())",
         },
-        "setValue": {
+        setValue: {
           "!doc": "数值操作",
-          "!type": "fn(name: string, operator: string, value: ?, prefix?: string)"
+          "!type":
+            "fn(name: string, operator: string, value: ?, prefix?: string)",
         },
-        "precompile": {
+        precompile: {
           "!doc": "预编辑事件",
-          "!type": "fn(data?: ?)"
+          "!type": "fn(data?: ?)",
         },
-        "vibrate": {
-          "!doc": "视野抖动<br/>例如:core.vibrate(); // 视野抖动1秒<br/>direction: 抖动方向;可填 horizontal(左右),vertical(上下),diagonal1(左上右下),diagonal2(左下右上)<br/>time: 抖动时长<br/>speed: 抖动速度<br/>power: 抖动幅度<br/>callback: 抖动平息后的回调函数,可选",
-          "!type": "fn(direction?: string, time?: number, speed?: number, power?: number, callback?: fn())"
+        vibrate: {
+          "!doc":
+            "视野抖动<br/>例如:core.vibrate(); // 视野抖动1秒<br/>direction: 抖动方向;可填 horizontal(左右),vertical(上下),diagonal1(左上右下),diagonal2(左下右上)<br/>time: 抖动时长<br/>speed: 抖动速度<br/>power: 抖动幅度<br/>callback: 抖动平息后的回调函数,可选",
+          "!type":
+            "fn(direction?: string, time?: number, speed?: number, power?: number, callback?: fn())",
         },
-        "confirmRestart": {
+        confirmRestart: {
           "!doc": "询问是否需要重新开始",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "battle": {
-          "!doc": "战斗,如果填写了坐标就会删除该点的敌人并触发战后事件<br/>例如:core.battle('greenSlime'); // 和从天而降的绿头怪战斗(如果打得过)<br/>id: 敌人id,必填<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>force: true表示强制战斗,可选<br/>callback: 回调函数,可选",
-          "!type": "fn(id: string, x?: number, y?: number, force?: bool, callback?: fn())"
+        battle: {
+          "!doc":
+            "战斗,如果填写了坐标就会删除该点的敌人并触发战后事件<br/>例如:core.battle('greenSlime'); // 和从天而降的绿头怪战斗(如果打得过)<br/>id: 敌人id,必填<br/>x: 敌人的横坐标,可选<br/>y: 敌人的纵坐标,可选<br/>force: true表示强制战斗,可选<br/>callback: 回调函数,可选",
+          "!type":
+            "fn(id: string, x?: number, y?: number, force?: bool, callback?: fn())",
         },
-        "follow": {
-          "!doc": "跟随<br/>name: 要跟随的一个合法的4x4的行走图名称,需要在全塔属性注册",
-          "!type": "fn(name: string)"
+        follow: {
+          "!doc":
+            "跟随<br/>name: 要跟随的一个合法的4x4的行走图名称,需要在全塔属性注册",
+          "!type": "fn(name: string)",
         },
-        "beforeBattle": {
+        beforeBattle: {
           "!doc": "战斗前触发的事件;返回false代表不进行战斗",
-          "!type": "fn(enemyId?: string, x?: number, y?: number) -> bool"
+          "!type": "fn(enemyId?: string, x?: number, y?: number) -> bool",
         },
-        "registerEvent": {
-          "!doc": "注册一个自定义事件<br/>type: 事件类型<br/>func: 事件的处理函数,可接受(data, x, y, prefix)参数<br/>data为事件内容,x和y为当前点坐标(可为null),prefix为当前点前缀",
-          "!type": "fn(type: string, func: fn(data: ?, x?: number, y?: number, prefix?: string))"
+        registerEvent: {
+          "!doc":
+            "注册一个自定义事件<br/>type: 事件类型<br/>func: 事件的处理函数,可接受(data, x, y, prefix)参数<br/>data为事件内容,x和y为当前点坐标(可为null),prefix为当前点前缀",
+          "!type":
+            "fn(type: string, func: fn(data: ?, x?: number, y?: number, prefix?: string))",
         },
-        "flyTo": {
+        flyTo: {
           "!doc": "飞往某一层",
-          "!type": "fn(toId?: string, callback?: fn()) -> bool"
+          "!type": "fn(toId?: string, callback?: fn()) -> bool",
         },
-        "afterGetItem": {
+        afterGetItem: {
           "!doc": "获得一个道具后的事件",
-          "!type": "fn(id?: string, x?: number, y?: number, isGentleClick?: bool)"
+          "!type":
+            "fn(id?: string, x?: number, y?: number, isGentleClick?: bool)",
         },
-        "doAction": {
-          "!doc": "执行下一个事件指令,常作为回调<br/>例如:core.setCurtain([0,0,0,1], null, null, core.doAction); // 事件中的原生脚本,配合勾选“不自动执行下一个事件”来达到此改变色调只持续到下次场景切换的效果",
-          "!type": "fn()"
+        doAction: {
+          "!doc":
+            "执行下一个事件指令,常作为回调<br/>例如:core.setCurtain([0,0,0,1], null, null, core.doAction); // 事件中的原生脚本,配合勾选“不自动执行下一个事件”来达到此改变色调只持续到下次场景切换的效果",
+          "!type": "fn()",
         },
-        "openBook": {
+        openBook: {
           "!doc": "点击怪物手册时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "save": {
+        save: {
           "!doc": "点击存档按钮时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "load": {
+        load: {
           "!doc": "点击读档按钮时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "getNextItem": {
-          "!doc": "轻按获得面前的物品或周围唯一物品<br/>noRoute: 若为true则不计入录像",
-          "!type": "fn(noRoute?: bool)"
+        getNextItem: {
+          "!doc":
+            "轻按获得面前的物品或周围唯一物品<br/>noRoute: 若为true则不计入录像",
+          "!type": "fn(noRoute?: bool)",
         },
-        "hasAsync": {
+        hasAsync: {
           "!doc": "当前是否有未处理完毕的异步事件(不包含动画和音效)",
-          "!type": "fn() -> bool"
+          "!type": "fn() -> bool",
         },
-        "stopAsync": {
+        stopAsync: {
           "!doc": "立刻停止所有正在进行的异步事件",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "openEquipbox": {
+        openEquipbox: {
           "!doc": "点击装备栏时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "recoverEvents": {
+        recoverEvents: {
           "!doc": "恢复一个事件",
-          "!type": "fn(data?: ?)"
+          "!type": "fn(data?: ?)",
         },
-        "setGlobalFlag": {
-          "!doc": "设置一个系统开关<br/>例如:core.setGlobalFlag('steelDoorWithoutKey', true); // 使全塔的所有铁门都不再需要钥匙就能打开<br/>name: 系统开关的英文名<br/>value: 开关的新值,您可以用!core.flags[name]简单地表示将此开关反转",
-          "!type": "fn(name: string, value: bool)"
+        setGlobalFlag: {
+          "!doc":
+            "设置一个系统开关<br/>例如:core.setGlobalFlag('steelDoorWithoutKey', true); // 使全塔的所有铁门都不再需要钥匙就能打开<br/>name: 系统开关的英文名<br/>value: 开关的新值,您可以用!core.flags[name]简单地表示将此开关反转",
+          "!type": "fn(name: string, value: bool)",
         },
-        "moveImage": {
-          "!doc": "移动一张图片并/或改变其透明度<br/>例如:core.moveImage(1, null, 0.5); // 1秒内把1号图片变为50%透明<br/>code: 图片编号<br/>to: 新的左上角坐标,省略表示原地改变透明度<br/>opacityVal: 新的透明度,省略表示不变<br/>time: 移动用时,单位为毫秒。不填视为1秒<br/>callback: 图片移动完毕后的回调函数,可选",
-          "!type": "fn(code: number, to?: [number], opacityVal?: number, moveMode?: string, time?: number, callback?: fn())"
+        moveImage: {
+          "!doc":
+            "移动一张图片并/或改变其透明度<br/>例如:core.moveImage(1, null, 0.5); // 1秒内把1号图片变为50%透明<br/>code: 图片编号<br/>to: 新的左上角坐标,省略表示原地改变透明度<br/>opacityVal: 新的透明度,省略表示不变<br/>time: 移动用时,单位为毫秒。不填视为1秒<br/>callback: 图片移动完毕后的回调函数,可选",
+          "!type":
+            "fn(code: number, to?: [number], opacityVal?: number, moveMode?: string, time?: number, callback?: fn())",
         },
-        "rotateImage": {
-          "!doc": "旋转一张图片<br/>code: 图片编号<br/>center: 旋转中心像素坐标(以屏幕为基准);不填视为图片本身中心<br/>angle: 旋转角度;正数为顺时针,负数为逆时针<br/>moveMode: 旋转模式<br/>time: 旋转用时,单位为毫秒。不填视为1秒<br/>callback: 图片旋转完毕后的回调函数,可选",
-          "!type": "fn(code: number, center?: [number], angle?: number, moveMode?: string, time?: number, callback?: fn())"
+        rotateImage: {
+          "!doc":
+            "旋转一张图片<br/>code: 图片编号<br/>center: 旋转中心像素坐标(以屏幕为基准);不填视为图片本身中心<br/>angle: 旋转角度;正数为顺时针,负数为逆时针<br/>moveMode: 旋转模式<br/>time: 旋转用时,单位为毫秒。不填视为1秒<br/>callback: 图片旋转完毕后的回调函数,可选",
+          "!type":
+            "fn(code: number, center?: [number], angle?: number, moveMode?: string, time?: number, callback?: fn())",
         },
-        "scaleImage": {
+        scaleImage: {
           "!doc": "放缩一张图片",
-          "!type": "fn(code: number, center?: [number], scale?: number, moveMode?: string, time?: number, callback?: fn())"
+          "!type":
+            "fn(code: number, center?: [number], scale?: number, moveMode?: string, time?: number, callback?: fn())",
         },
-        "moveTextBox": {
+        moveTextBox: {
           "!doc": "移动对话框",
-          "!type": "fn(code: number, loc: [number], relative?: bool, moveMode?: string, time?: number, callback?: fn())"
+          "!type":
+            "fn(code: number, loc: [number], relative?: bool, moveMode?: string, time?: number, callback?: fn())",
         },
-        "clearTextBox": {
+        clearTextBox: {
           "!doc": "清除对话框",
-          "!type": "fn(code: number)"
+          "!type": "fn(code: number)",
         },
-        "openSettings": {
+        openSettings: {
           "!doc": "点击设置按钮时的操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "afterPushBox": {
+        afterPushBox: {
           "!doc": "推箱子后的事件",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "unregisterSystemEvent": {
+        unregisterSystemEvent: {
           "!doc": "注销一个系统事件",
-          "!type": "fn(type: string)"
+          "!type": "fn(type: string)",
         },
-        "trigger": {
-          "!doc": "触发(x,y)点的系统事件;会执行该点图块的script属性,同时支持战斗(会触发战后)、道具(会触发道具后)、楼层切换等等<br/>callback: 执行完毕的回调函数<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(x?: number, y?: number, callback?: fn())"
+        trigger: {
+          "!doc":
+            "触发(x,y)点的系统事件;会执行该点图块的script属性,同时支持战斗(会触发战后)、道具(会触发道具后)、楼层切换等等<br/>callback: 执行完毕的回调函数<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type": "fn(x?: number, y?: number, callback?: fn())",
         },
-        "restart": {
+        restart: {
           "!doc": "重新开始游戏;此函数将回到标题页面",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "doEvent": {
+        doEvent: {
           "!doc": "执行一个自定义事件",
-          "!type": "fn(data?: ?, x?: number, y?: number, prefix?: string)"
+          "!type": "fn(data?: ?, x?: number, y?: number, prefix?: string)",
         },
-        "win": {
+        win: {
           "!doc": "游戏获胜事件",
-          "!type": "fn(reason?: string, norank?: bool, noexit?: bool)"
+          "!type": "fn(reason?: string, norank?: bool, noexit?: bool)",
         },
-        "setGlobalAttribute": {
+        setGlobalAttribute: {
           "!doc": "设置全塔属性",
-          "!type": "fn(name: string, value: string)"
+          "!type": "fn(name: string, value: string)",
         },
-        "setNameMap": {
+        setNameMap: {
           "!doc": "设置文件别名",
-          "!type": "fn(name: string, value?: string)"
+          "!type": "fn(name: string, value?: string)",
         },
-        "setTextAttribute": {
+        setTextAttribute: {
           "!doc": "设置剧情文本的属性",
-          "!type": "fn(data: ?)"
+          "!type": "fn(data: ?)",
         },
-        "openToolbox": {
+        openToolbox: {
           "!doc": "点击工具栏时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "setVolume": {
-          "!doc": "调节bgm的音量<br/>例如:core.setVolume(0, 100, core.jumpHero); // 0.1秒内淡出bgm,然后主角原地跳跃半秒<br/>value: 新的音量,为0或不大于1的正数。注意系统设置中是这个值的平方根的十倍<br/>time: 渐变用时,单位为毫秒。不填或小于100毫秒都视为0<br/>callback: 渐变完成后的回调函数,可选",
-          "!type": "fn(value: number, time?: number, callback?: fn())"
+        setVolume: {
+          "!doc":
+            "调节bgm的音量<br/>例如:core.setVolume(0, 100, core.jumpHero); // 0.1秒内淡出bgm,然后主角原地跳跃半秒<br/>value: 新的音量,为0或不大于1的正数。注意系统设置中是这个值的平方根的十倍<br/>time: 渐变用时,单位为毫秒。不填或小于100毫秒都视为0<br/>callback: 渐变完成后的回调函数,可选",
+          "!type": "fn(value: number, time?: number, callback?: fn())",
         },
-        "pushEventLoc": {
+        pushEventLoc: {
           "!doc": "将当前点坐标入栈",
-          "!type": "fn(x?: number, y?: number, floorId?: string) -> bool"
+          "!type": "fn(x?: number, y?: number, floorId?: string) -> bool",
         },
-        "openKeyBoard": {
+        openKeyBoard: {
           "!doc": "点击虚拟键盘时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "insertCommonEvent": {
-          "!doc": "插入一个公共事件<br/>例如:core.insertCommonEvent('加点事件', [3]);<br/>name: 公共事件名;如果公共事件不存在则直接忽略<br/>args: 参数列表,为一个数组,将依次赋值给 flag:arg1, flag:arg2, ...<br/>x: 新的当前点横坐标,可选<br/>y: 新的当前点纵坐标,可选<br/>callback: 新的回调函数,可选<br/>addToLast: 插入的位置,true表示插入到末尾,否则插入到开头",
-          "!type": "fn(name?: string, args?: [?], x?: number, y?: number, callback?: fn(), addToLast?: bool)"
+        insertCommonEvent: {
+          "!doc":
+            "插入一个公共事件<br/>例如:core.insertCommonEvent('加点事件', [3]);<br/>name: 公共事件名;如果公共事件不存在则直接忽略<br/>args: 参数列表,为一个数组,将依次赋值给 flag:arg1, flag:arg2, ...<br/>x: 新的当前点横坐标,可选<br/>y: 新的当前点纵坐标,可选<br/>callback: 新的回调函数,可选<br/>addToLast: 插入的位置,true表示插入到末尾,否则插入到开头",
+          "!type":
+            "fn(name?: string, args?: [?], x?: number, y?: number, callback?: fn(), addToLast?: bool)",
         },
-        "hideImage": {
-          "!doc": "隐藏一张图片<br/>例如:core.hideImage(1, 1000, core.jumpHero); // 1秒内淡出1号图片,然后主角原地跳跃半秒<br/>code: 图片编号<br/>time: 淡出时间,单位为毫秒<br/>callback: 图片完全消失后的回调函数,可选",
-          "!type": "fn(code: number, time?: number, callback?: fn())"
+        hideImage: {
+          "!doc":
+            "隐藏一张图片<br/>例如:core.hideImage(1, 1000, core.jumpHero); // 1秒内淡出1号图片,然后主角原地跳跃半秒<br/>code: 图片编号<br/>time: 淡出时间,单位为毫秒<br/>callback: 图片完全消失后的回调函数,可选",
+          "!type": "fn(code: number, time?: number, callback?: fn())",
         },
-        "visitFloor": {
+        visitFloor: {
           "!doc": "到达某楼层",
-          "!type": "fn(floorId?: string)"
+          "!type": "fn(floorId?: string)",
         },
-        "openQuickShop": {
+        openQuickShop: {
           "!doc": "点击快捷商店按钮时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
         },
-        "afterBattle": {
+        afterBattle: {
           "!doc": "战斗结束后触发的事件",
-          "!type": "fn(enemyId?: string, x?: number, y?: number)"
+          "!type": "fn(enemyId?: string, x?: number, y?: number)",
         },
-        "pushBox": {
+        pushBox: {
           "!doc": "推箱子",
-          "!type": "fn(data?: ?)"
+          "!type": "fn(data?: ?)",
         },
-        "autoEventExecuted": {
+        autoEventExecuted: {
           "!doc": "当前是否执行过某个自动事件",
-          "!type": "fn(symbol?: string, value?: ?) -> bool"
+          "!type": "fn(symbol?: string, value?: ?) -> bool",
         },
-        "onSki": {
+        onSki: {
           "!doc": "当前是否在冰上",
-          "!type": "fn(number?: number) -> bool"
+          "!type": "fn(number?: number) -> bool",
         },
-        "showImage": {
-          "!doc": "显示一张图片<br/>例如:core.showImage(1, core.material.images.images['winskin.png'], [0,0,128,128], [0,0,416,416], 0.5, 1000); // 裁剪winskin.png的最左边128×128px,放大到铺满整个视野,1秒内淡入到50%透明,编号为1<br/>code: 图片编号,为不大于50的正整数,加上100后就是对应画布层的z值,较大的会遮罩较小的,注意色调层的z值为125,UI层为140<br/>image: 图片文件名(可以是全塔属性中映射前的中文名)或图片对象(见上面的例子)<br/>sloc: 一行且至多四列的数组,表示从原图裁剪的左上角坐标和宽高,可选<br/>loc: 一行且至多四列的数组,表示图片在视野中的左上角坐标和宽高,可选<br/>opacityVal: 不透明度,为小于1的正数。不填视为1<br/>time: 淡入时间,单位为毫秒。不填视为0<br/>callback: 图片完全显示出来后的回调函数,可选",
-          "!type": "fn(code: number, image: string|image, sloc?: [number], loc?: [number], opacityVal?: number, time?: number, callback?: fn())"
+        showImage: {
+          "!doc":
+            "显示一张图片<br/>例如:core.showImage(1, core.material.images.images['winskin.png'], [0,0,128,128], [0,0,416,416], 0.5, 1000); // 裁剪winskin.png的最左边128×128px,放大到铺满整个视野,1秒内淡入到50%透明,编号为1<br/>code: 图片编号,为不大于50的正整数,加上100后就是对应画布层的z值,较大的会遮罩较小的,注意色调层的z值为125,UI层为140<br/>image: 图片文件名(可以是全塔属性中映射前的中文名)或图片对象(见上面的例子)<br/>sloc: 一行且至多四列的数组,表示从原图裁剪的左上角坐标和宽高,可选<br/>loc: 一行且至多四列的数组,表示图片在视野中的左上角坐标和宽高,可选<br/>opacityVal: 不透明度,为小于1的正数。不填视为1<br/>time: 淡入时间,单位为毫秒。不填视为0<br/>callback: 图片完全显示出来后的回调函数,可选",
+          "!type":
+            "fn(code: number, image: string|image, sloc?: [number], loc?: [number], opacityVal?: number, time?: number, callback?: fn())",
         },
-        "getItem": {
-          "!doc": "获得道具并提示,如果填写了坐标就会删除该点的该道具<br/>例如:core.getItem('book'); // 获得敌人手册并提示<br/>id: 道具id,必填<br/>num: 获得的数量,不填视为1,填了就别填坐标了<br/>x: 道具的横坐标,可选<br/>y: 道具的纵坐标,可选<br/>callback: 回调函数,可选",
-          "!type": "fn(id: string, num?: number, x?: number, y?: number, callback?: fn())"
+        getItem: {
+          "!doc":
+            "获得道具并提示,如果填写了坐标就会删除该点的该道具<br/>例如:core.getItem('book'); // 获得敌人手册并提示<br/>id: 道具id,必填<br/>num: 获得的数量,不填视为1,填了就别填坐标了<br/>x: 道具的横坐标,可选<br/>y: 道具的纵坐标,可选<br/>callback: 回调函数,可选",
+          "!type":
+            "fn(id: string, num?: number, x?: number, y?: number, callback?: fn())",
         },
-        "registerSystemEvent": {
-          "!doc": "注册一个系统事件<br/>type: 事件名<br/>func: 为事件的处理函数,可接受(data,callback)参数",
-          "!type": "fn(type: string, func: fn(data?: ?, callback?: fn()))"
+        registerSystemEvent: {
+          "!doc":
+            "注册一个系统事件<br/>type: 事件名<br/>func: 为事件的处理函数,可接受(data,callback)参数",
+          "!type": "fn(type: string, func: fn(data?: ?, callback?: fn()))",
         },
-        "startGame": {
-          "!doc": "开始新游戏<br/>例如:core.startGame('咸鱼乱撞', 0, ''); // 开始一局咸鱼乱撞难度的新游戏,随机种子为0<br/>hard: 难度名,会显示在左下角(横屏)或右下角(竖屏)<br/>seed: 随机种子,相同的种子保证了录像的可重复性<br/>route: 经由base64压缩后的录像,用于从头开始的录像回放<br/>callback: 回调函数,可选",
-          "!type": "fn(hard: string, seed: number, route: string, callback?: fn())"
+        startGame: {
+          "!doc":
+            "开始新游戏<br/>例如:core.startGame('咸鱼乱撞', 0, ''); // 开始一局咸鱼乱撞难度的新游戏,随机种子为0<br/>hard: 难度名,会显示在左下角(横屏)或右下角(竖屏)<br/>seed: 随机种子,相同的种子保证了录像的可重复性<br/>route: 经由base64压缩后的录像,用于从头开始的录像回放<br/>callback: 回调函数,可选",
+          "!type":
+            "fn(hard: string, seed: number, route: string, callback?: fn())",
         },
-        "doSystemEvent": {
+        doSystemEvent: {
           "!doc": "执行一个系统事件",
-          "!type": "fn(type: string, data?: ?, callback?: fn())"
+          "!type": "fn(type: string, data?: ?, callback?: fn())",
         },
-        "resetGame": {
+        resetGame: {
           "!doc": "初始化游戏",
-          "!type": "fn(hero?: ?, hard?: ?, floorId?: string, maps?: ?, values?: ?)"
+          "!type":
+            "fn(hero?: ?, hard?: ?, floorId?: string, maps?: ?, values?: ?)",
         },
-        "setFloorInfo": {
-          "!doc": "设置一项楼层属性并刷新状态栏<br/>例如:core.setFloorInfo('ratio', 2, 'MT0'); // 把主塔0层的血瓶和宝石变为双倍效果<br/>name: 要修改的属性名<br/>values: 属性的新值。<br/>floorId: 楼层id,不填视为当前层<br/>prefix: 独立开关前缀,一般不需要",
-          "!type": "fn(name: string, values: ?, floorId?: string, prefix?: string)"
+        setFloorInfo: {
+          "!doc":
+            "设置一项楼层属性并刷新状态栏<br/>例如:core.setFloorInfo('ratio', 2, 'MT0'); // 把主塔0层的血瓶和宝石变为双倍效果<br/>name: 要修改的属性名<br/>values: 属性的新值。<br/>floorId: 楼层id,不填视为当前层<br/>prefix: 独立开关前缀,一般不需要",
+          "!type":
+            "fn(name: string, values: ?, floorId?: string, prefix?: string)",
         },
-        "openDoor": {
-          "!doc": "开门(包括三种基础墙)<br/>例如:core.openDoor(0, 0, true, core.jumpHero); // 打开左上角的门,需要钥匙,然后主角原地跳跃半秒<br/>x: 门的横坐标<br/>y: 门的纵坐标<br/>needKey: true表示需要钥匙,会导致机关门打不开<br/>callback: 门完全打开后或打不开时的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(x: number, y: number, needKey?: bool, callback?: fn())"
+        openDoor: {
+          "!doc":
+            "开门(包括三种基础墙)<br/>例如:core.openDoor(0, 0, true, core.jumpHero); // 打开左上角的门,需要钥匙,然后主角原地跳跃半秒<br/>x: 门的横坐标<br/>y: 门的纵坐标<br/>needKey: true表示需要钥匙,会导致机关门打不开<br/>callback: 门完全打开后或打不开时的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type": "fn(x: number, y: number, needKey?: bool, callback?: fn())",
         },
-        "setEnemy": {
-          "!doc": "设置一项敌人属性并计入存档<br/>例如:core.setEnemy('greenSlime', 'def', 0); // 把绿头怪的防御设为0<br/>id: 敌人id<br/>name: 属性的英文缩写<br/>value: 属性的新值,可选<br/>operator: 运算操作符如+=,可选<br/>prefix: 独立开关前缀,一般不需要,下同",
-          "!type": "fn(id: string, name: string, value: ?, operator?: string, prefix?: string)"
+        setEnemy: {
+          "!doc":
+            "设置一项敌人属性并计入存档<br/>例如:core.setEnemy('greenSlime', 'def', 0); // 把绿头怪的防御设为0<br/>id: 敌人id<br/>name: 属性的英文缩写<br/>value: 属性的新值,可选<br/>operator: 运算操作符如+=,可选<br/>prefix: 独立开关前缀,一般不需要,下同",
+          "!type":
+            "fn(id: string, name: string, value: ?, operator?: string, prefix?: string)",
         },
-        "setEnemyOnPoint": {
-          "!doc": "设置某个点的敌人属性。如果该点不是怪物,则忽略此函数。<br/>例如:core.setEnemyOnPoint(3, 5, null, 'atk', 100, '+='); // 仅将(3,5)点怪物的攻击力加100。",
-          "!type": "fn(x: number, y: number, floorId?: string, name: string, value: ?, operator?: string, prefix?: string)"
+        setEnemyOnPoint: {
+          "!doc":
+            "设置某个点的敌人属性。如果该点不是怪物,则忽略此函数。<br/>例如:core.setEnemyOnPoint(3, 5, null, 'atk', 100, '+='); // 仅将(3,5)点怪物的攻击力加100。",
+          "!type":
+            "fn(x: number, y: number, floorId?: string, name: string, value: ?, operator?: string, prefix?: string)",
         },
-        "resetEnemyOnPoint": {
+        resetEnemyOnPoint: {
           "!doc": "重置某个点的怪物属性",
-          "!type": "fn(x: number, y: number, floorId?: string)"
+          "!type": "fn(x: number, y: number, floorId?: string)",
         },
-        "moveEnemyOnPoint": {
+        moveEnemyOnPoint: {
           "!doc": "将某个点已经设置的敌人属性移动到其他点",
-          "!type": "fn(fromX: number, fromY: number, toX: number, toY: number, floorId?: string)"
+          "!type":
+            "fn(fromX: number, fromY: number, toX: number, toY: number, floorId?: string)",
         },
-        "autoEventExecuting": {
+        autoEventExecuting: {
           "!doc": "当前是否在执行某个自动事件",
-          "!type": "fn(symbol?: string, value?: ?) -> bool"
+          "!type": "fn(symbol?: string, value?: ?) -> bool",
         },
-        "checkAutoEvents": {
+        checkAutoEvents: {
           "!doc": "检测自动事件",
-          "!type": "fn()"
+          "!type": "fn()",
         },
-        "showGif": {
-          "!doc": "绘制一张动图或擦除所有动图<br/>例如:core.showGif(); // 擦除所有动图<br/>name: 动图文件名,可以是全塔属性中映射前的中文名<br/>x: 动图在视野中的左上角横坐标<br/>y: 动图在视野中的左上角纵坐标",
-          "!type": "fn(name?: string, x?: number, y?: number)"
+        showGif: {
+          "!doc":
+            "绘制一张动图或擦除所有动图<br/>例如:core.showGif(); // 擦除所有动图<br/>name: 动图文件名,可以是全塔属性中映射前的中文名<br/>x: 动图在视野中的左上角横坐标<br/>y: 动图在视野中的左上角纵坐标",
+          "!type": "fn(name?: string, x?: number, y?: number)",
         },
-        "unregisterEvent": {
+        unregisterEvent: {
           "!doc": "注销一个自定义事件",
-          "!type": "fn(type: string)"
+          "!type": "fn(type: string)",
         },
-        "jumpHero": {
-          "!doc": "主角跳跃,跳跃勇士。ex和ey为目标点的坐标,可以为null表示原地跳跃。time为总跳跃时间。<br/>例如:core.jumpHero(); // 主角原地跳跃半秒<br/>ex: 跳跃后的横坐标<br/>ey: 跳跃后的纵坐标<br/>time: 跳跃时长,单位为毫秒。不填视为半秒<br/>callback: 跳跃完毕后的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(ex?: number, ey?: number, time?: number, callback?: fn())"
+        jumpHero: {
+          "!doc":
+            "主角跳跃,跳跃勇士。ex和ey为目标点的坐标,可以为null表示原地跳跃。time为总跳跃时间。<br/>例如:core.jumpHero(); // 主角原地跳跃半秒<br/>ex: 跳跃后的横坐标<br/>ey: 跳跃后的纵坐标<br/>time: 跳跃时长,单位为毫秒。不填视为半秒<br/>callback: 跳跃完毕后的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type":
+            "fn(ex?: number, ey?: number, time?: number, callback?: fn())",
         },
-        "closeDoor": {
-          "!doc": "关门,目标点必须为空地<br/>例如:core.closeDoor(0, 0, 'yellowWall', core.jumpHero); // 在左上角关掉一堵黄墙,然后主角原地跳跃半秒<br/>x: 横坐标<br/>y: 纵坐标<br/>id: 门的id,也可以用三种基础墙<br/>callback: 门完全关上后的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(x: number, y: number, id: string, callback?: fn())"
+        closeDoor: {
+          "!doc":
+            "关门,目标点必须为空地<br/>例如:core.closeDoor(0, 0, 'yellowWall', core.jumpHero); // 在左上角关掉一堵黄墙,然后主角原地跳跃半秒<br/>x: 横坐标<br/>y: 纵坐标<br/>id: 门的id,也可以用三种基础墙<br/>callback: 门完全关上后的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type": "fn(x: number, y: number, id: string, callback?: fn())",
         },
-        "eventMoveHero": {
-          "!doc": "强制移动主角(包括后退),这个函数的作者已经看不懂这个函数了<br/>例如:core.eventMoveHero(['forward'], 125, core.jumpHero); // 主角强制前进一步,用时1/8秒,然后主角原地跳跃半秒<br/>steps: 步伐数组,注意后退时跟随者的行为会很难看<br/>time: 每步的用时,单位为毫秒。0或不填则取主角的移速,如果后者也不存在就取0.1秒<br/>callback: 移动完毕后的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(steps: [step], time?: number, callback?: fn())"
+        eventMoveHero: {
+          "!doc":
+            "强制移动主角(包括后退),这个函数的作者已经看不懂这个函数了<br/>例如:core.eventMoveHero(['forward'], 125, core.jumpHero); // 主角强制前进一步,用时1/8秒,然后主角原地跳跃半秒<br/>steps: 步伐数组,注意后退时跟随者的行为会很难看<br/>time: 每步的用时,单位为毫秒。0或不填则取主角的移速,如果后者也不存在就取0.1秒<br/>callback: 移动完毕后的回调函数,可选<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type": "fn(steps: [step], time?: number, callback?: fn())",
         },
-        "changeFloor": {
-          "!doc": "场景切换<br/>例如:core.changeFloor('MT0'); // 传送到主塔0层,主角坐标和朝向不变,黑屏时间取用户定义的值<br/>floorId: 传送的目标地图id,可以填':before'和':next'分别表示楼下或楼上<br/>stair: 传送的位置<br/>heroLoc: 传送的坐标;会覆盖stair<br/>time: 传送的黑屏时间,单位为毫秒;不填为用户设置值<br/>callback: 传送的回调函数<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
-          "!type": "fn(floorId: string, stair?: string, heroLoc?: {x?: number, y?: number, direction?: string}, time?: number, callback?: fn())"
+        changeFloor: {
+          "!doc":
+            "场景切换<br/>例如:core.changeFloor('MT0'); // 传送到主塔0层,主角坐标和朝向不变,黑屏时间取用户定义的值<br/>floorId: 传送的目标地图id,可以填':before'和':next'分别表示楼下或楼上<br/>stair: 传送的位置<br/>heroLoc: 传送的坐标;会覆盖stair<br/>time: 传送的黑屏时间,单位为毫秒;不填为用户设置值<br/>callback: 传送的回调函数<br/>【异步脚本,请勿在脚本中直接调用(而是使用对应的事件),否则可能导致录像出错】",
+          "!type":
+            "fn(floorId: string, stair?: string, heroLoc?: {x?: number, y?: number, direction?: string}, time?: number, callback?: fn())",
         },
-        "getCommonEvent": {
+        getCommonEvent: {
           "!doc": "获得一个公共事件",
-          "!type": "fn(name: string) -> [?]"
+          "!type": "fn(name: string) -> [?]",
         },
-        "lose": {
+        lose: {
           "!doc": "游戏失败事件",
-          "!type": "fn(reason?: string)"
+          "!type": "fn(reason?: string)",
         },
-        "gameOver": {
-          "!doc": "游戏结束<br/>例如:core.gameOver(); // 游戏失败<br/>ending: 结局名,省略表示失败<br/>fromReplay: true表示在播放录像,可选<br/>norank: true表示不计入榜单,可选",
-          "!type": "fn(ending?: string, fromReplay?: bool, norank?: bool)"
+        gameOver: {
+          "!doc":
+            "游戏结束<br/>例如:core.gameOver(); // 游戏失败<br/>ending: 结局名,省略表示失败<br/>fromReplay: true表示在播放录像,可选<br/>norank: true表示不计入榜单,可选",
+          "!type": "fn(ending?: string, fromReplay?: bool, norank?: bool)",
         },
-        "useFly": {
+        useFly: {
           "!doc": "点击楼层传送器时的打开操作",
-          "!type": "fn(fromUserAction?: bool)"
+          "!type": "fn(fromUserAction?: bool)",
+        },
+        tryUseItem: {
+          "!doc":
+            "尝试使用一个道具<br/>例如:core.tryUseItem('pickaxe'); // 尝试使用破墙镐<br/>itemId: 道具id,其中敌人手册、传送器和飞行器会被特殊处理",
+          "!type": "fn(itemId: string)",
         },
-        "tryUseItem": {
-          "!doc": "尝试使用一个道具<br/>例如:core.tryUseItem('pickaxe'); // 尝试使用破墙镐<br/>itemId: 道具id,其中敌人手册、传送器和飞行器会被特殊处理",
-          "!type": "fn(itemId: string)"
-        }
       },
-      "plugin": {
+      plugin: {
         "!doc": "插件编写中内置了一些常用的插件。",
-        "drawLight": {
-          "!doc": "绘制一段灯光效果<br/>name:必填,要绘制到的画布名;可以是一个系统画布,或者是个自定义画布;如果不存在则创建<br/>color:可选,只能是一个0~1之间的数,为不透明度的值。不填则默认为0.9。<br/>lights:可选,一个数组,定义了每个独立的灯光。其中每一项是三元组 [x,y,r] x和y分别为该灯光的横纵坐标,r为该灯光的半径。<br/>lightDec:可选,0到1之间,光从多少百分比才开始衰减(在此范围内保持全亮),不设置默认为0。比如lightDec为0.5代表,每个灯光部分内圈50%的范围全亮,50%以后才开始快速衰减。<br/>例如:core.plugin.drawLight('test', 0.2, [[25,11,46,0.1]]); // 创建一个test图层,不透明度0.2,其中在(25,11)点存在一个半径为46的灯光效果,灯光中心不透明度0.1。<br/>core.plugin.drawLight('test2', 0.9, [[25,11,46],[105,121,88],[301,221,106]]); // 创建test2图层,且存在三个灯光效果,分别是中心(25,11)半径46,中心(105,121)半径88,中心(301,221)半径106。",
-          "!type": "fn(name: string|CanvasRenderingContext2D, color?: number, lights?: [[number]], lightDec?: number)"
+        drawLight: {
+          "!doc":
+            "绘制一段灯光效果<br/>name:必填,要绘制到的画布名;可以是一个系统画布,或者是个自定义画布;如果不存在则创建<br/>color:可选,只能是一个0~1之间的数,为不透明度的值。不填则默认为0.9。<br/>lights:可选,一个数组,定义了每个独立的灯光。其中每一项是三元组 [x,y,r] x和y分别为该灯光的横纵坐标,r为该灯光的半径。<br/>lightDec:可选,0到1之间,光从多少百分比才开始衰减(在此范围内保持全亮),不设置默认为0。比如lightDec为0.5代表,每个灯光部分内圈50%的范围全亮,50%以后才开始快速衰减。<br/>例如:core.plugin.drawLight('test', 0.2, [[25,11,46,0.1]]); // 创建一个test图层,不透明度0.2,其中在(25,11)点存在一个半径为46的灯光效果,灯光中心不透明度0.1。<br/>core.plugin.drawLight('test2', 0.9, [[25,11,46],[105,121,88],[301,221,106]]); // 创建test2图层,且存在三个灯光效果,分别是中心(25,11)半径46,中心(105,121)半径88,中心(301,221)半径106。",
+          "!type":
+            "fn(name: string|CanvasRenderingContext2D, color?: number, lights?: [[number]], lightDec?: number)",
         },
-        "openShop": {
-          "!doc": "打开一个全局商店<br/>shopId: 要开启的商店ID<br/>noRoute: 打开行为是否不计入录像",
-          "!type": "fn(shopId: string, noRoute?: bool)"
+        openShop: {
+          "!doc":
+            "打开一个全局商店<br/>shopId: 要开启的商店ID<br/>noRoute: 打开行为是否不计入录像",
+          "!type": "fn(shopId: string, noRoute?: bool)",
         },
-        "isShopVisited": {
+        isShopVisited: {
           "!doc": "某个全局商店是否被访问过",
-          "!type": "fn(id: string) -> bool"
+          "!type": "fn(id: string) -> bool",
         },
-        "listShopIds": {
+        listShopIds: {
           "!doc": "列出所有应当显示的快捷商店列表",
-          "!type": "fn() -> [string]"
+          "!type": "fn() -> [string]",
         },
-        "canOpenShop": {
+        canOpenShop: {
           "!doc": "当前能否打开某个商店",
-          "!type": "fn(id: string) -> bool"
+          "!type": "fn(id: string) -> bool",
         },
-        "setShopVisited": {
+        setShopVisited: {
           "!doc": "设置某个商店的访问状态",
-          "!type": "fn(id: string, visited?: bool)"
+          "!type": "fn(id: string, visited?: bool)",
         },
-        "canUseQuickShop": {
-          "!doc": "当前能否使用某个快捷商店<br/>如果返回一个字符串,则代表不能,返回的字符串作为不能的提示;返回null表示可以使用",
-          "!type": "fn(id: string) -> string"
+        canUseQuickShop: {
+          "!doc":
+            "当前能否使用某个快捷商店<br/>如果返回一个字符串,则代表不能,返回的字符串作为不能的提示;返回null表示可以使用",
+          "!type": "fn(id: string) -> string",
         },
-        "removeMaps": {
-          "!doc": "删除某一些楼层;删除后不会存入存档,不可浏览地图也不可飞到。<br/>fromId: 开始删除的楼层ID<br/>toId: 删除到的楼层编号;可选,不填则视为fromId<br/>例如:core.removeMaps(\"MT1\", \"MT300\") 删除MT1~MT300之间的全部层<br/>core.removeMaps(\"MT10\") 只删除MT10层",
-          "!type": "fn(fromId: string, toId?: string)"
+        removeMaps: {
+          "!doc":
+            '删除某一些楼层;删除后不会存入存档,不可浏览地图也不可飞到。<br/>fromId: 开始删除的楼层ID<br/>toId: 删除到的楼层编号;可选,不填则视为fromId<br/>例如:core.removeMaps("MT1", "MT300") 删除MT1~MT300之间的全部层<br/>core.removeMaps("MT10") 只删除MT10层',
+          "!type": "fn(fromId: string, toId?: string)",
         },
-        "resumeMaps": {
-          "!doc": "恢复某一些被删除楼层。<br/>fromId: 开始恢复的楼层ID<br/>toId: 恢复到的楼层编号;可选,不填则视为fromId<br/>例如:core.resumeMaps(\"MT1\", \"MT300\") 恢复MT1~MT300之间的全部层<br/>core.resumeMaps(\"MT10\") 只删恢复MT10层",
-          "!type": "fn(fromId: string, toId?: string)"
+        resumeMaps: {
+          "!doc":
+            '恢复某一些被删除楼层。<br/>fromId: 开始恢复的楼层ID<br/>toId: 恢复到的楼层编号;可选,不填则视为fromId<br/>例如:core.resumeMaps("MT1", "MT300") 恢复MT1~MT300之间的全部层<br/>core.resumeMaps("MT10") 只删恢复MT10层',
+          "!type": "fn(fromId: string, toId?: string)",
         },
-        "autoRemoveMaps": {
+        autoRemoveMaps: {
           "!doc": "根据楼层分区信息自动砍层与恢复",
-          "!type": "fn(floorId: string)"
+          "!type": "fn(floorId: string)",
         },
-        "openItemShop": {
+        openItemShop: {
           "!doc": "打开一个道具商店",
-          "!type": "fn(itemShopId: string)"
-        }
-      }
+          "!type": "fn(itemShopId: string)",
+        },
+      },
     },
-    "lzw_encode": {
+    lzw_encode: {
       "!doc": "LZW压缩算法",
       "!url": "https://gist.github.com/revolunet/843889",
-      "!type": "fn(s: string) -> string"
+      "!type": "fn(s: string) -> string",
     },
-    "lzw_decode": {
+    lzw_decode: {
       "!doc": "LZW解压缩算法",
       "!url": "https://gist.github.com/revolunet/843889",
-      "!type": "fn(s: string) -> string"
+      "!type": "fn(s: string) -> string",
     },
-    "hero": {
+    hero: {
       "!type": "heroStatus",
       "!doc": "勇士信息,为 core.status.hero 的简写",
     },
-    "flags": {
+    flags: {
       "!type": "flag",
       "!doc": "游戏中用到的变量,为 core.status.hero.flags 的简写",
-    }
-  }
-];
\ No newline at end of file
+    },
+  },
+];
diff --git a/_server/MotaAction.g4 b/_server/MotaAction.g4
index 10b3ad5..b4a2a2f 100644
--- a/_server/MotaAction.g4
+++ b/_server/MotaAction.g4
@@ -567,7 +567,7 @@ doorInfo_m
 
 /* doorInfo_m
 tooltip : 开门信息
-default : [160, 'door.mp3', 'door.mp3']
+default : [160, 'door.opus', 'door.opus']
 helpUrl : /_docs/#/instruction
 EvalString_0 = EvalString_0 && (', "openSound": "' + EvalString_0 + '"');
 EvalString_1 = EvalString_1 && (', "closeSound": "' + EvalString_1 + '"');
@@ -837,6 +837,14 @@ action
     |   insert_1_s
     |   insert_2_s
     |   exit_s
+    |   setanimate_s
+    |   deleteanimate_s
+    |   playanimate_s
+    |   clearanimate_s
+    |   animateloop_s
+    |   animatemove_s
+    |   animatereverse_s
+    |   animatepause_s
     |   setBlock_s
     |   turnBlock_s
     |   showFloorImg_s
@@ -854,6 +862,8 @@ action
     |   waitAsync_s
     |   stopAsync_s
     |   op_s
+    |   setmusics_s
+    |   introAndLoop_s
     |   battle_s
     |   battle_1_s
     |   openDoor_s
@@ -871,11 +881,11 @@ action
     |   disableShop_s
     |   follow_s
     |   unfollow_s
-    |   setanimate_s
-    |   deleteanimate_s
-    |   playanimate_s
-    |   clearanimate_s
     |   animate_s
+    |   moveAnimate_s
+    |   animateResize_s
+    |   pauseAnimate_s
+    |   remuseAnimate_s
     |   animate_1_s
     |   stopAnimate_s
     |   vibrate_s
@@ -903,6 +913,8 @@ action
     |   resumeBgm_s
     |   loadBgm_s
     |   freeBgm_s
+    |   drawbackground_s
+    |   clearbackground_s
     |   playSound_s
     |   playSound_1_s
     |   stopSound_s
@@ -963,8 +975,7 @@ action
     |   animationDrawable_s
     |   over_s
     |   overlist_s
-    |   playStereo_s
-    |   moveStereo_s
+    |   cgtextList_s
     |   pass_s
     ;
 
@@ -1071,6 +1082,98 @@ if (block.isCollapsed() || !block.isEnabled() || pos || Int_0 || Bool_0) {
 return code+',\n';
 */;
 
+
+animationDrawable_s
+    :   '帧动画(剧情画面)' '总显示帧数' IntString '底色' ColorString? Colour '底色不透明度' IntString? BGNL?Newline
+        '图片序列(同一帧后面覆盖先前的,默认起始帧为0,结束帧为最后一帧)'BGNL?Newline
+        '(剪裁区域不填写为全图,绘制区域不填写为全画面)'BGNL?Newline
+        '(不透明度100为不透明,默认为不透明,结束不透明度默认与开始透明度相同)'BGNL?Newline
+        animationDrawableimage+? Newline
+        '音频序列(到达对应帧进行播放)'BGNL?Newline
+        animationDrawablesound+? Newline
+    
+
+/* animationDrawable_s
+tooltip : animationDrawable:帧动画图片叠加
+helpUrl : /_docs/#/instruction
+default : [30,"0, 0, 0",'rgb(0, 0, 0)',""]
+allImages : ['EvalString_0']
+colour : this.imageColor
+if (IntString_1&&(IntString_1 < 0||IntString_1>100)) throw new Error('透明度范围为0-100,0为透明,100为不透明,不填默认为不透明')
+
+ColorString_0 = ColorString_0 ? (', "color": ['+ColorString_0+']') : '';
+IntString_1 = IntString_1 ? (', "globalAlpha": '+IntString_1+'') : '';
+var imageList=animationDrawableimage_0?',"imageList": [\n'+animationDrawableimage_0.slice(0,-1)+'\n]':''
+var soundList=animationDrawablesound_0?',"soundList": [\n'+animationDrawablesound_0.slice(0,-1)+'\n]':''
+var code = '{"type": "animationDrawable", "allFarme": '+IntString_0+ColorString_0+IntString_1+imageList+soundList+'},\n';
+return code;
+*/;
+
+animationDrawableList
+    : animationDrawableimage
+    | animationDrawablesound
+    | animationDrawabletextEmpty;
+
+animationDrawableimage
+    : '图片' EvalString? '起始帧' IntString? '起始不透明度'  IntString? BGNL? Newline 
+    '剪裁坐标cx' IntString? 'cy' IntString? '宽' IntString? '高' IntString? '绘制坐标x'IntString? 'y' IntString? '宽' IntString? '高' IntString?'角度'IntString? BGNL? Newline
+    '结束帧' IntString? '结束不透明度'  IntString? '剪裁坐标cx' IntString? 'cy' IntString? '宽' IntString? '高' IntString? '绘制坐标x'IntString? 'y' IntString? '宽' IntString? '高' IntString?'角度'IntString? Newline
+/* animationDrawableimage
+tooltip : 帧动画图片列表
+helpUrl : /_docs/#/instruction
+default : ["","","","","","","","","","","","","","","","","","","","","","",""]
+colour : this.subColor
+allImages : ['EvalString_0']
+if (IntString_1&&(IntString_1 < 0||IntString_1>100)) throw new Error('不透明度范围为0-100,0为透明,100为不透明,不填默认为不透明')
+if (IntString_11&&(IntString_12 < 0||IntString_12>100)) throw new Error('不透明度范围为0-100,0为透明,100为不透明,不填默认为不透明')
+IntString_0 = IntString_0 ? (', "beforefarme": '+IntString_0+'') : '';
+IntString_1 = IntString_1 ? (', "globalAlpha": '+IntString_1+'') : '';
+IntString_2 = IntString_2 ? (', "cx": '+IntString_2+'') : '';
+IntString_3 = IntString_3 ? (', "cy": '+IntString_3+'') : '';
+IntString_4 = IntString_4 ? (', "cw": '+IntString_4+'') : '';
+IntString_5 = IntString_5 ? (', "ch": '+IntString_5+'') : '';
+IntString_6 = IntString_6 ? (', "x": '+IntString_6+'') : '';
+IntString_7 = IntString_7 ? (', "y": '+IntString_7+'') : '';
+IntString_8 = IntString_8 ? (', "w": '+IntString_8+'') : '';
+IntString_9 = IntString_9 ? (', "h": '+IntString_9+'') : '';
+
+IntString_10 = IntString_10 ? (', "angle": '+IntString_10+'') : '';
+IntString_11 = IntString_11 ? (', "afterfarme": '+IntString_11+'') : '';
+IntString_12 = IntString_12 ? (', "aglobalAlpha": '+IntString_12+'') : '';
+IntString_13 = IntString_13 ? (', "acx": '+IntString_13+'') : '';
+IntString_14 = IntString_14 ? (', "acy": '+IntString_14+'') : '';
+IntString_15 = IntString_15 ? (', "acw": '+IntString_15+'') : '';
+IntString_16 = IntString_16 ? (', "ach": '+IntString_16+'') : '';
+IntString_17 = IntString_17 ?  (', "ax": '+IntString_17+'')  : '';
+IntString_18 = IntString_18 ? (', "ay": '+IntString_18+'') : '';
+IntString_19 = IntString_19 ? (', "aw": '+IntString_19+'') : '';
+IntString_20 = IntString_20 ? (', "ah": '+IntString_20+'') : '';
+IntString_21 = IntString_21 ? (', "aangle": '+IntString_21+'') : '';
+return '{ "image":"'+EvalString_0+'"'+IntString_0+IntString_1+IntString_2+IntString_3+IntString_4+IntString_5+IntString_6+IntString_7+IntString_8+IntString_9+IntString_10+IntString_11+IntString_12+IntString_13+IntString_14+IntString_15+IntString_16+IntString_17+IntString_18+IntString_19+IntString_20+IntString_21+'},'
+*/;
+
+animationDrawablesound
+    : '音效' EvalString? '起始帧' IntString '是否停止其他音效'Bool?  Newline 
+ 
+/* animationDrawablesound
+tooltip : 帧动画配音
+helpUrl : /_docs/#/instruction
+default : ["",0,false]
+colour : this.subColor
+allSounds : ['EvalString_0']
+
+return '{ "sound":"'+EvalString_0+'","startfarme":'+IntString_0+',"stopbefore":'+Bool_0+'},'
+*/;
+
+animationDrawabletextEmpty
+    :   Newline
+    
+/* animationDrawabletextEmpty
+var code = [];
+return code;
+*/;
+
+
 over_s
     :   '旁白文字' ':' EvalString BGNL? Newline
     '背景' EvalString? '回忆滤镜' Bool? '持续帧数' Int? '渐入渐出帧数' Int? '音频文件' EvalString? BGNL? Newline
@@ -1123,22 +1226,32 @@ overtextEmpty
 var code = [];
 return code;
 */;
+cgtextList_s
+    :    '切换剧情文本库' EvalString  Newline
+    
+
+/* cgtextList_s
+tooltip : cgtextList:设置cg剧情文本库
+helpUrl : /_docs/#/instruction
+default : ["chapter0"]
+var code = '{"type": "cgtextList", "textList":"'+EvalString_0+'"},\n';
+return code;
+*/;
 
 cgtext_s
-    :    '背景' EvalString? '回忆滤镜' Bool? '移除对话框' Bool? '头像' EvalString?'名字' EvalString?'坐标PX' Number'打字间隔' Int? BGNL? Newline
-     '自动等待时长' Int '音频文件(需在全塔属性——使用音效注册)'EvalString? BGNL? Newline
-     EvalString? BGNL? Newline textcgDrawingList+? Newline
+    : '剧情cg文本 移除对话框' Bool? '剧情库序列' Int '头像' EvalString?'坐标PX' Number'打字间隔' Int? BGNL? Newline
+     '自动等待时长' Int  BGNL? Newline
+      textcgDrawingList+? Newline
     
 
 /* cgtext_s
 tooltip : cgtext:显示一段包含cg的文字(剧情)
 helpUrl : /_docs/#/instruction
-allImages : ['EvalString_0','EvalString_1']
-allSounds : ['EvalString_3']
-default : ["bg_5043.webp",false,"face_050445.webp",false,"菲奥奈",-300,0,2000,"","这句话显示在对话框内",[{ "name":"tati_050145a.webp" , "px": 100,"filter":false }]]
-var head ='{ "name": "'+EvalString_1+'", "px": '+Number_0+' }'
+allImages : ['EvalString_0']
+default : [false,0,"face_050445.webp",-300,0,2000,"","这句话显示在对话框内",[{ "name":"tati_050145a.webp" , "px": 100,"filter":false }]]
+var head ='{ "name": "'+EvalString_0+'", "px": '+Number_0+' }'
 var list=',"bodyList": [\n'+textcgDrawingList_0.slice(0,-1)+'\n]'
-var code = '{"type": "cgtext", "bg":"'+EvalString_0+'","memory":'+Bool_0+',"WindowSkin":'+Bool_1+',"head":'+head+' ,"name":"'+EvalString_2+'","time":'+Int_0+',"wait":'+Int_1+',"sound":"'+EvalString_3+'","text": "'+EvalString_4+'"'+list+' },\n';
+var code = '{"type": "cgtext","WindowSkin":'+Bool_0+',"head":'+head+' ,"index":"'+Int_0+'","time":'+Int_1+',"wait":'+Int_2+list+' },\n';
 return code;
 */;
 textcgDrawingList
@@ -1955,17 +2068,46 @@ var code = '{"type": "changebg", "img1": "'+EvalString_0+'","memory1":'+Bool_0+'
 return code;
 */;
 
+drawbackground_s
+    :   '绘制背景' EvalString? '回忆滤镜' Bool? Newline
+    
+
+/* drawbackground_s
+tooltip : drawbackground 绘制剧情背景
+helpUrl : /_docs/#/instruction
+default : ["",false]
+colour : this.soundColor
+allImages : ['EvalString_0']
+var code = '{"type": "drawbackground", "image": "'+EvalString_0+'","memory":'+Bool_0+'},\n';
+return code;
+*/;
+
+clearbackground_s
+    :   '清除cg背景' Newline
+    
+
+/* clearbackground_s
+tooltip : clearbackground: 剧情背景清除
+helpUrl : /_docs/#/instruction
+default : []
+colour : this.soundColor
+
+var code = '{"type": "clearbackground"},\n';
+return code;
+*/;
+
 drawWarning_s
-    :   '警告坐标x' PosString 'y' PosString '文字大小'Int '警告内容' EvalString? '副标题' EvalString? '不播放音效' Bool? Newline
+    :   '警告坐标x' PosString 'y' PosString '缩放倍率'EvalString '文字大小'Int '警告内容' EvalString? '副标题' EvalString? '不播放音效' Bool? Newline
     
 
 /* drawWarning_s
 tooltip : drawWarning: 绘制warning警告并拉进镜头,本事件为异步事件,下一事件将在3100ms后执行
 helpUrl : /_docs/#/instruction
-default : [0, 0,24,"这里是提示内容","绿色史莱姆",false]
+default : [0, 0,2,24,"这里是提示内容","绿色史莱姆",false]
 colour : this.soundColor
 selectPoint : ["PosString_0", "PosString_1"]
-var code = '{"type": "drawWarning", "x": '+PosString_0+', "y": '+PosString_1+',"size":'+Int_0+',"text": "'+EvalString_0+'", "text2": "'+EvalString_1+'", "warning": '+Bool_0+'},\n';
+if(!EvalString_1||!/^(0|([1-9][0-9]*))(\.[\d]+)?$/.test(EvalString_0))throw new Error("此项仅能填写小数或整数");
+var code = '{"type": "drawWarning", "x": '+PosString_0+', "y": '+PosString_1+',"large":'+EvalString_0+',"size":'+Int_0+',"text": "'+EvalString_1+'", "text2": "'+EvalString_2+'", "warning": '+Bool_0+'},\n';
 return code;
 */;
 
@@ -2230,6 +2372,69 @@ var code = '{"type": "vibrate", "direction": "'+Vibrate_List_0+'", "time": '+Int
 return code;
 */;
 
+animate_s
+    :   '显示动画' EvalString '位置' 'x' PosString? 'y' PosString? '相对窗口坐标' Bool '不等待执行完毕' Bool Newline
+    
+
+/* animate_s
+tooltip : animate:显示动画,位置填hero或者1,2形式的位置,或者不填代表当前事件点
+helpUrl : /_docs/#/instruction
+default : ["zone","","",false,false]
+allAnimates : ['EvalString_0']
+material : ["./project/animates/", "EvalString_0"]
+menu : [['选择位置', 'editor_blockly.selectPoint(block, ["PosString_0", "PosString_1"])']]
+colour : this.soundColor
+var loc = PosString_0&&PosString_1?(', "loc": ['+PosString_0+','+PosString_1+']'):'';
+Bool_0 = Bool_0?', "alignWindow": true':'';
+Bool_1 = Bool_1?', "async": true':'';
+var code = '{"type": "animate", "name": "'+EvalString_0+'"'+loc+Bool_0+Bool_1+'},\n';
+return code;
+*/;
+
+animate_1_s
+    :   '显示动画并跟随角色' EvalString '不等待执行完毕' Bool Newline
+    
+
+/* animate_1_s
+tooltip : animate:显示动画并跟随角色
+helpUrl : /_docs/#/instruction
+default : ["zone",false]
+allAnimates : ['EvalString_0']
+material : ["./project/animates/", "EvalString_0"]
+colour : this.soundColor
+Bool_0 = Bool_0?', "async": true':'';
+var code = '{"type": "animate", "name": "'+EvalString_0+'", "loc": "hero"'+Bool_0+'},\n';
+return code;
+*/;
+
+
+animatereverse_s
+    :   '调整正在播放的帧动画/特效  编号'IntString?'倒放'Bool'(不填编号为对所有正在进行的帧动画/特效进行操作)'
+
+/* animatereverse_s
+tooltip : animatereverse:调整正在播放的帧动画(倒放)
+helpUrl : /_docs/#/instruction
+default : ["",false]
+colour : this.imageColor
+IntString_0 = IntString_0 ? (', "id": '+IntString_0+'') : ''
+var code = '{"type": "animatereverse"'+IntString_0+',"reverse":'+Bool_0+'},\n';
+return code;
+*/;
+
+animatepause_s
+    :   '调整正在播放的帧动画/特效  编号'IntString?'暂停'Bool'(不填编号为对所有正在进行的帧动画/特效进行操作)'
+
+/* animatepause_s
+tooltip :  animatereverse:调整正在播放的帧动画(暂停)
+helpUrl : /_docs/#/instruction
+default : ["",false]
+colour : this.imageColor
+IntString_0 = IntString_0 ? (', "id": '+IntString_0+'') : ''
+var code = '{"type": "animatepause"'+IntString_0+',"pause":'+Bool_0+'},\n';
+return code;
+*/;
+
+
 setanimate_s
     :   '新建 帧动画/特效' '名称' EvalString '参照点偏移像素x' IntString? 'y' IntString? '动画大小 宽' IntString '高' IntString '总帧数' IntString BGNL?Newline
         '图片序列(同一帧后面覆盖先前的,默认起始帧为0,结束帧为最后一帧)'BGNL?Newline
@@ -2331,82 +2536,148 @@ return code;
 */;
 
 playanimate_s
-    :   '播放 帧动画/特效' '名称' EvalString '像素x'  IntString? 'y' IntString? '跟随勇士' Bool 'x方向缩放' EvalString? 'y方向缩放'EvalString?  Newline
+    :   '播放 帧动画/特效' '名称' EvalString '编号'IntString?'像素x'  IntString? 'y' IntString? '跟随勇士' Bool 'x方向缩放' EvalString? 'y方向缩放'EvalString?'循环'Bool'倒放'Bool  Newline
 
 /* playanimate_s
 tooltip : playanimate:播放帧动画,选择跟随勇士后x、y将失效改为勇士中心坐标
 helpUrl : /_docs/#/instruction
-default : ["zone","","",false,"",""]
-
+default : ["zone","","","",false,1,1,false,false]
+previewBlock : true
 colour : this.imageColor
-IntString_0 = IntString_0 ? (', "x": '+IntString_0+'') : '';
-IntString_1 = IntString_1 ? (', "y": '+IntString_1+'') : '';
+IntString_0 = IntString_0 ? (', "id": '+IntString_0+'') : '';
+IntString_1 = IntString_1 ? (', "x": '+IntString_1+'') : '';
+IntString_2 = IntString_2 ? (', "y": '+IntString_2+'') : '';
 if(EvalString_1&&!/^(0|([1-9][0-9]*))(\.[\d]+)?$/.test(EvalString_1))throw new Error("此项仅能填写小数、整数或不填");
 if(EvalString_2&&!/^(0|([1-9][0-9]*))(\.[\d]+)?$/.test(EvalString_2))throw new Error("此项仅能填写小数、整数或不填");
 EvalString_1 = EvalString_1 ? (', "scalex": '+EvalString_1+'') : '';
 EvalString_2 = EvalString_2 ? (', "scaley": '+EvalString_2+'') : '';
-var code = '{"type": "playanimate", "name": "'+EvalString_0+'"'+IntString_0+IntString_1+',"hero":'+Bool_0+EvalString_1+EvalString_2+'},\n';
+var code = '{"type": "playanimate", "name": "'+EvalString_0+'"'+IntString_0+IntString_1+IntString_2+',"hero":'+Bool_0+EvalString_1+EvalString_2+',"loop":'+Bool_1+',"reverse":'+Bool_2+'},\n';
 return code;
 */;
 
 clearanimate_s
-    :   '清空正在播放的帧动画/特效'
+    :   '停止正在播放的帧动画/特效  编号'IntString?'(不填编号为对所有正在进行的帧动画/特效进行操作)'
 
 /* clearanimate_s
 tooltip : clearanimate:清空正在播放的帧动画
 helpUrl : /_docs/#/instruction
-
+default : [""]
 colour : this.imageColor
-
-var code = '{"type": "clearanimate"},\n';
+IntString_0 = IntString_0 ? (', "id": '+IntString_0+'') : '';
+var code = '{"type": "clearanimate"'+IntString_0+'},\n';
 return code;
 */;
 
-animate_s
-    :   '显示动画' EvalString '位置' 'x' PosString? 'y' PosString? '相对窗口坐标' Bool '不等待执行完毕' Bool Newline
-    
+animateloop_s
+    :   '调整正在播放的帧动画/特效  编号'IntString?'循环'Bool '(不填编号为对所有正在进行的帧动画/特效进行操作)'
 
-/* animate_s
-tooltip : animate:显示动画,位置填hero或者1,2形式的位置,或者不填代表当前事件点
+/* animateloop_s
+tooltip : animateloop:调整正在播放的帧动画(循环)
 helpUrl : /_docs/#/instruction
-default : ["zone","","",false,false]
-allAnimates : ['EvalString_0']
-material : ["./project/animates/", "EvalString_0"]
-menu : [['选择位置', 'editor_blockly.selectPoint(block, ["PosString_0", "PosString_1"])']]
-colour : this.soundColor
-var loc = PosString_0&&PosString_1?(', "loc": ['+PosString_0+','+PosString_1+']'):'';
-Bool_0 = Bool_0?', "alignWindow": true':'';
-Bool_1 = Bool_1?', "async": true':'';
-var code = '{"type": "animate", "name": "'+EvalString_0+'"'+loc+Bool_0+Bool_1+'},\n';
+default : ["",false]
+colour : this.imageColor
+IntString_0 = IntString_0 ? (', "id": '+IntString_0+'') : ''
+var code = '{"type": "animateloop"'+IntString_0+',"loop":'+Bool_0+'},\n';
 return code;
 */;
 
-animate_1_s
-    :   '显示动画并跟随角色' EvalString '不等待执行完毕' Bool Newline
+animatemove_s
+    :   '移动正在播放的帧动画/特效  编号'IntString'目标像素x'PosString'y'PosString Absolute_List'移动时长'PosString'移动模式'MoveMode2_List? '(仅可对有编号的非跟随勇士帧动画/特效使用)'
+
+/* animatemove_s
+tooltip : animateloop:调整正在播放的帧动画(循环)
+helpUrl : /_docs/#/instruction
+default : [0,0,0,false,1000,'']
+previewBlock : true
+colour : this.imageColor
+IntString_0 = IntString_0 ? (', "id": '+IntString_0+'') : ''
+MoveMode2_List_0=MoveMode2_List_0?(', "style": "'+MoveMode2_List_0+'"') : ''
+var code = '{"type": "animatemove"'+IntString_0+',"px":'+PosString_0+',"py":'+PosString_1+',"relative":'+Absolute_List_0+',"time":'+PosString_2+MoveMode2_List_0+'},\n';
+return code;
+*/;
+
+
+moveAnimate_s
+    :   '移动正在播放的帧动画/特效  编号'IntString'目标像素x'PosString'y'PosString Absolute_List'移动时长'PosString'移动模式'MoveMode2_List? '(仅可对有编号的非跟随勇士动画使用)'
+
+/* moveAnimate_s
+tooltip : animateloop:调整正在播放的帧动画(循环)
+helpUrl : /_docs/#/instruction
+default : [0,0,0,false,1000,'']
+previewBlock : true
+colour : this.soundColor
+IntString_0 = IntString_0 ? (', "id": '+IntString_0+'') : ''
+MoveMode2_List_0=MoveMode2_List_0?(', "style": "'+MoveMode2_List_0+'"') : ''
+var code = '{"type": "moveAnimate"'+IntString_0+',"px":'+PosString_0+',"py":'+PosString_1+',"relative":'+Absolute_List_0+',"time":'+PosString_2+MoveMode2_List_0+'},\n';
+return code;
+*/;
+
+
+animateResize_s
+ :   '显示动画(60Fps)' EvalString '编号'PosString? '中心像素' 'x' PosString? 'y' PosString?'以勇士为中心' Bool'倒放' Bool'循环'Bool'不等待执行完毕' Bool Newline
     
 
-/* animate_1_s
-tooltip : animate:显示动画并跟随角色
+/* animateResize_s
+tooltip : animateResize:显示动画并选择是否以勇士为中心及倒放(以勇士为中心时,中心像素坐标无效)
 helpUrl : /_docs/#/instruction
-default : ["zone",false]
+default : ["zone","","","",false,false,false,false]
 allAnimates : ['EvalString_0']
+previewBlock : true
 material : ["./project/animates/", "EvalString_0"]
 colour : this.soundColor
-Bool_0 = Bool_0?', "async": true':'';
-var code = '{"type": "animate", "name": "'+EvalString_0+'", "loc": "hero"'+Bool_0+'},\n';
+
+PosString_0 = PosString_0?', "id":'+PosString_0:'';
+Bool_0 = Bool_0?', "hero": true':'';
+if(!Bool_0){
+    if(PosString_1===""||PosString_2==="")throw new Error("中心像素坐标或以勇士为中心必须填写一项");
+    
+    Bool_0=',"centerX":'+PosString_1+', "centerY":'+PosString_2
+}
+if(Bool_2&&!Bool_3 )throw new Error("循环必须与不等待执行完毕同时开启,否则将陷入死循环")
+Bool_1 = Bool_1?', "reverse": true':'';
+Bool_2 = Bool_2?', "loop": true':'';
+Bool_3 = Bool_3?', "async": true':'';
+var code = '{"type": "animateResize", "name": "'+EvalString_0+'"'+PosString_0+Bool_0+Bool_1+Bool_2+Bool_3+'},\n';
+return code;
+*/;
+
+pauseAnimate_s
+ :   '暂停动画编号'PosString?'(不填写编号为暂停所有)' Newline
+
+/* pauseAnimate_s
+tooltip : pauseAnimate:暂停动画(不填写编号为暂停所有)
+helpUrl : /_docs/#/instruction
+default : ['']
+colour : this.soundColor
+PosString_0 = PosString_0?', "id":'+PosString_0:'';
+var code = '{"type": "pauseAnimate"'+PosString_0+'},\n';
+return code;
+*/;
+
+remuseAnimate_s
+ :   '继续动画编号'PosString?'(不填写编号为继续所有)'  Newline
+
+/* remuseAnimate_s
+tooltip : remuseAnimate:继续动画(不填写编号为继续所有)
+helpUrl : /_docs/#/instruction
+default : ['']
+colour : this.soundColor
+PosString_0 = PosString_0?', "id":'+PosString_0:'';
+var code = '{"type": "remuseAnimate"'+PosString_0+'},\n';
 return code;
 */;
 
 stopAnimate_s
-    :   '停止所有动画' '执行动画回调' Bool Newline
+    :   '停止动画编号'PosString?'(不填写编号为继续所有)' '执行动画回调' Bool Newline
 
 /* stopAnimate_s
-tooltip : stopAnimate:停止所有动画
+tooltip : stopAnimate:停止动画(不填写编号为继续所有)
 helpUrl : /_docs/#/instruction
-default : [false]
-colour : this.imageColor
+default : ["",false]
+colour : this.soundColor
+PosString_0 = PosString_0?', "id":'+PosString_0:'';
 Bool_0 = Bool_0?', "doCallback": true':'';
-var code = '{"type": "stopAnimate"'+Bool_0+'},\n';
+var code = '{"type": "stopAnimate"'+PosString_0+Bool_0+'},\n';
 return code;
 */;
 
@@ -2612,6 +2883,23 @@ var code = '{"type": "setq"'+IdString_0+'},\n';
 return code;
 */;
 
+introAndLoop_s
+    :   '前奏音乐文件'EvalString? '前奏音乐文件播放时长(可填小数,单位为秒)'EvalString'循环音乐文件'EvalString?  Newline
+
+/* introAndLoop_s
+tooltip : introAndLoop:设置任务目标楼层(显示在小地图)
+helpUrl : /_docs/#/instruction
+default : ["",1,""]
+allBgms : ['EvalString_0','EvalString_2']
+colour :this.imageColor
+if(EvalString_1&&!/^(0|([1-9][0-9]*))(\.[\d]+)?$/.test(EvalString_1))throw new Error("此项仅能填写小数或整数,单位为秒");
+EvalString_0 = EvalString_0 ? (', "intro": "'+EvalString_0+'"') : '';
+EvalString_1= EvalString_1 ? (', "time": '+EvalString_1+'') : '';
+EvalString_2 = EvalString_2 ? (', "loop": "'+EvalString_2+'"') : '';
+var code = '{"type": "introAndLoop"'+EvalString_0+EvalString_1+EvalString_2+'},\n';
+return code;
+*/;
+
 
 setcgs_s
     :   'cg回廊获取cg 文件名'EvalString?'不填为清空cg获取记录'  Newline
@@ -2621,12 +2909,26 @@ tooltip : setcgs:cg回廊中的cg获取
 helpUrl : /_docs/#/instruction
 default : [""]
 allImages : ['EvalString_0']
-colour : this.imageColor
+colour : this.soundColor
 EvalString_0 = EvalString_0 ? (', "img": "'+EvalString_0+'"') : '';
 var code = '{"type": "setcgs"'+EvalString_0+'},\n';
 return code;
 */;
 
+setmusics_s
+    :   '音乐鉴赏music 文件名'EvalString?'不填为清空music获取记录'  Newline
+
+/* setmusics_s
+tooltip : setmusics:音乐鉴赏中的隐藏music获取
+helpUrl : /_docs/#/instruction
+default : [""]
+allBgms : ['EvalString_0']
+colour : this.soundColor
+EvalString_0 = EvalString_0 ? (', "bgm": "'+EvalString_0+'"') : '';
+var code = '{"type": "setmusics"'+EvalString_0+'},\n';
+return code;
+*/;
+
 scaleImage_s
     :   '图片放缩' '图片编号' NInt '中心点像素' 'x' PosString? 'y' PosString? '移动方式' MoveMode_List BGNL? '放缩比例' Number '动画时间' Int '不等待执行完毕' Bool Newline
        
@@ -2647,95 +2949,6 @@ return code;
 */;
 
 
-animationDrawable_s
-    :   '帧动画' '总显示帧数' IntString '底色' ColorString? Colour '底色不透明度' IntString? BGNL?Newline
-        '图片序列(同一帧后面覆盖先前的,默认起始帧为0,结束帧为最后一帧)'BGNL?Newline
-        '(剪裁区域不填写为全图,绘制区域不填写为全画面)'BGNL?Newline
-        '(不透明度100为不透明,默认为不透明,结束不透明度默认与开始透明度相同)'BGNL?Newline
-        animationDrawableimage+? Newline
-        '音频序列(到达对应帧进行播放)'BGNL?Newline
-        animationDrawablesound+? Newline
-    
-
-/* animationDrawable_s
-tooltip : animationDrawable:帧动画图片叠加
-helpUrl : /_docs/#/instruction
-default : [30,"0, 0, 0",'rgb(0, 0, 0)',""]
-allImages : ['EvalString_0']
-colour : this.imageColor
-if (IntString_1&&(IntString_1 < 0||IntString_1>100)) throw new Error('透明度范围为0-100,0为透明,100为不透明,不填默认为不透明')
-
-ColorString_0 = ColorString_0 ? (', "color": ['+ColorString_0+']') : '';
-IntString_1 = IntString_1 ? (', "globalAlpha": '+IntString_1+'') : '';
-var imageList=animationDrawableimage_0?',"imageList": [\n'+animationDrawableimage_0.slice(0,-1)+'\n]':''
-var soundList=animationDrawablesound_0?',"soundList": [\n'+animationDrawablesound_0.slice(0,-1)+'\n]':''
-var code = '{"type": "animationDrawable", "allFarme": '+IntString_0+ColorString_0+IntString_1+imageList+soundList+'},\n';
-return code;
-*/;
-
-animationDrawableList
-    : animationDrawableimage
-    | animationDrawablesound
-    | animationDrawabletextEmpty;
-
-animationDrawableimage
-    : '图片' EvalString? '起始帧' IntString? '起始不透明度'  IntString? BGNL? Newline 
-    '剪裁坐标cx' IntString? 'cy' IntString? '宽' IntString? '高' IntString? '绘制坐标x'IntString? 'y' IntString? '宽' IntString? '高' IntString? BGNL? Newline
-    '结束帧' IntString? '结束不透明度'  IntString? '剪裁坐标cx' IntString? 'cy' IntString? '宽' IntString? '高' IntString? '绘制坐标x'IntString? 'y' IntString? '宽' IntString? '高' IntString? Newline
-/* animationDrawableimage
-tooltip : 帧动画图片列表
-helpUrl : /_docs/#/instruction
-default : ["","","","","","","","","","","","","","","","","","","","",""]
-colour : this.subColor
-allImages : ['EvalString_0']
-if (IntString_1&&(IntString_1 < 0||IntString_1>100)) throw new Error('不透明度范围为0-100,0为透明,100为不透明,不填默认为不透明')
-if (IntString_11&&(IntString_11 < 0||IntString_11>100)) throw new Error('不透明度范围为0-100,0为透明,100为不透明,不填默认为不透明')
-IntString_0 = IntString_0 ? (', "beforefarme": '+IntString_0+'') : '';
-IntString_1 = IntString_1 ? (', "globalAlpha": '+IntString_1+'') : '';
-IntString_2 = IntString_2 ? (', "cx": '+IntString_2+'') : '';
-IntString_3 = IntString_3 ? (', "cy": '+IntString_3+'') : '';
-IntString_4 = IntString_4 ? (', "cw": '+IntString_4+'') : '';
-IntString_5 = IntString_5 ? (', "ch": '+IntString_5+'') : '';
-IntString_6 = IntString_6 ? (', "x": '+IntString_6+'') : '';
-IntString_7 = IntString_7 ? (', "y": '+IntString_7+'') : '';
-IntString_8 = IntString_8 ? (', "w": '+IntString_8+'') : '';
-IntString_9 = IntString_9 ? (', "h": '+IntString_9+'') : '';
-
-IntString_10 = IntString_10 ? (', "afterfarme": '+IntString_10+'') : '';
-IntString_11 = IntString_11 ? (', "aglobalAlpha": '+IntString_11+'') : '';
-IntString_12 = IntString_12 ? (', "acx": '+IntString_12+'') : '';
-IntString_13 = IntString_13 ? (', "acy": '+IntString_13+'') : '';
-IntString_14 = IntString_14 ? (', "acw": '+IntString_14+'') : '';
-IntString_15 = IntString_15 ? (', "ach": '+IntString_15+'') : '';
-IntString_16 = IntString_16 ? (', "ax": '+IntString_16+'') : '';
-IntString_17 = IntString_17 ? (', "ay": '+IntString_17+'') : '';
-IntString_18 = IntString_18 ? (', "aw": '+IntString_18+'') : '';
-IntString_19 = IntString_19 ? (', "ah": '+IntString_19+'') : '';
-
-return '{ "image":"'+EvalString_0+'"'+IntString_0+IntString_1+IntString_2+IntString_3+IntString_4+IntString_5+IntString_6+IntString_7+IntString_8+IntString_9+IntString_10+IntString_11+IntString_12+IntString_13+IntString_14+IntString_15+IntString_16+IntString_17+IntString_18+IntString_19+'},'
-*/;
-
-animationDrawablesound
-    : '音效' EvalString? '起始帧' IntString '是否停止其他音效'Bool?  Newline 
- 
-/* animationDrawablesound
-tooltip : 帧动画配音
-helpUrl : /_docs/#/instruction
-default : ["",0,false]
-colour : this.subColor
-allSounds : ['EvalString_0']
-
-return '{ "sound":"'+EvalString_0+'","startfarme":'+IntString_0+',"stopbefore":'+Bool_0+'},'
-*/;
-
-animationDrawabletextEmpty
-    :   Newline
-    
-/* animationDrawabletextEmpty
-var code = [];
-return code;
-*/;
-
 setCurtain_0_s
     :   '更改画面色调' ColorString Colour '动画时间' IntString? BGNL? Newline '渐变方式' MoveMode_List '持续到下一个本事件' Bool '不等待执行完毕' Bool Newline
     
@@ -2787,7 +3000,7 @@ return code;
 */;
 
 setWeather_s
-    :   '更改天气' Weather_List '强度' Int '持续到下个本事件' Bool Newline
+    :   '更改天气' Weather_List '强度' Int  Newline
     
 
 /* setWeather_s
@@ -2796,8 +3009,8 @@ helpUrl : /_docs/#/instruction
 default : [null,1,true]
 colour : this.soundColor
 if(Int_0<1 || Int_0>10) throw new Error('天气的强度等级, 在1-10之间');
-Bool_0 = Bool_0 ? ', "keep": true' : ''
-var code = '{"type": "setWeather", "name": "'+Weather_List_0+'", "level": '+Int_0+Bool_0+'},\n';
+
+var code = '{"type": "setWeather", "name": "'+Weather_List_0+'", "level": '+Int_0+'},\n';
 if(Weather_List_0===''||Weather_List_0==='null'||Weather_List_0==null)code = '{"type": "setWeather"},\n';
 return code;
 */;
@@ -3045,38 +3258,7 @@ var code = '{"type": "playSound", "name": "'+EvalString_0+'"'+Bool_0+IntString_0
 return code;
 */;
 
-playStereo_s
-    :   '播放音效(立体音)' EvalString '左声道音量' Int '右声道音量' Int '启用立体音' Bool? Newline
-    
 
-/* playStereo_s
-tooltip : playSound: 播放音效(立体音)
-helpUrl : /_docs/#/instruction
-default : ["item.mp3",1,1,true]
-colour : this.imageColor
-allSounds : ['EvalString_0']
-material : ["./project/sounds/", "EvalString_0"]
-
-var code = '{"type": "playStereo", "name": "'+EvalString_0+'", "left": '+Int_0+', "right": '+Int_1+', "split": '+Bool_0+'},\n';
-return code;
-*/;
-
-moveStereo_s
-    :   '播放渐变音效(立体音)' EvalString '左声道音量' Int '右声道音量' Int '启用立体音' Bool? BGNL? Newline
-    '左声道目标音量' Int '右声道目标音量' Int '渐变时长'Int Newline
-    
-
-/* moveStereo_s
-tooltip : playSound: 播放渐变音效(立体音)
-helpUrl : /_docs/#/instruction
-default : ["item.mp3",1,1,true,10,10,1000]
-colour : this.imageColor
-allSounds : ['EvalString_0']
-material : ["./project/sounds/", "EvalString_0"]
-
-var code = '{"type": "moveStereo", "name": "'+EvalString_0+'", "left": '+Int_0+', "right": '+Int_1+', "split": '+Bool_0+', "leftTo": '+Int_2+', "rightTo": '+Int_3+', "time": '+Int_4+'},\n';
-return code;
-*/;
 
 playSound_1_s
     :   '播放系统音效' NameMap_List '停止之前音效' Bool? '音调' IntString? '等待播放完毕' Bool? Newline
@@ -4517,6 +4699,10 @@ B_0_List
 B_1_List
     :   '不改变'|'设为粗体'|'取消粗体'
     /*B_1_List ['null','true','false']*/;
+    
+Absolute_List
+    :'绝对模式'|'相对模式'
+    /*Absolute_List ['false','true']*/;
 
 Bg_Fg_List
     :   '背景层'|'前景层'
@@ -4612,8 +4798,8 @@ IdString
     ;
 
 FixedId_List
-    :   '生命'|'生命上限'|'攻击'|'防御'|'护盾'|'黄钥匙'|'蓝钥匙'|'红钥匙'|'金币'|'经验'|'魔力'|'魔力上限'|'当前横坐标'|'当前纵坐标'|'当前朝向'|'攻击增益'|'防御增益'|'护盾增益'
-    /*FixedId_List ['status:hp','status:hpmax','status:atk','status:def','status:mdef','item:yellowKey','item:blueKey','item:redKey','status:money','status:exp','status:mana','status:manamax','status:x','status:y','status:direction','buff:atk','buff:def','buff:mdef']*/;
+    :   '生命'|'生命上限'|'攻击'|'防御'|'法强'|'魔攻比例'|'护盾比例'|'法抗比例'|'黄钥匙'|'蓝钥匙'|'红钥匙'|'金币'|'经验'|'魔力'|'魔力上限'|'当前横坐标'|'当前纵坐标'|'当前朝向'|'攻击增益'|'防御增益'|'法强增益'|'魔攻增益'|'法抗增益'|'护盾增益'
+    /*FixedId_List ['status:hp','status:hpmax','status:atk','status:def','status:spell','status:matk','status:mhp','status:mdef','item:yellowKey','item:blueKey','item:redKey','status:money','status:exp','status:mana','status:manamax','status:x','status:y','status:direction','buff:atk','buff:def','buff:spell','buff:matk','buff:mdef','buff:mhp']*/;
 
 Id_List
     :   '变量' | '状态' | '物品' | '增益' | '独立开关' | '临时变量' |'全局存储'
@@ -4628,8 +4814,8 @@ EnemyPoint_List
     /*EnemyPoint_List ['hp','atk','def','money','exp','point','name']*/;
 
 Equip_List
-    :   '生命'|'生命上限'|'攻击'|'防御'|'法强'|'魔力'|'魔力上限'|'速度'
-    /*Equip_List ['hp','hpmax','atk','def','mdef','mana','manamax','speed']*/;
+    :   '生命'|'生命上限'|'攻击'|'防御'|'法强'|'魔攻比例'|'护盾比例'|'法抗百分比'|'魔力'|'魔力上限'|'速度'
+    /*Equip_List ['hp','hpmax','atk','def','spell','matk','mhp','mdef','mana','manamax','speed']*/;
 
 Key_List
     :   '黄钥匙'|'蓝钥匙'|'红钥匙'|'绿钥匙'|'铁门钥匙'
@@ -4647,6 +4833,10 @@ MoveMode_List
     :   '匀速移动'|'缓入快出'|'快入缓出'|'缓入缓出'|'随机'
     /*MoveMode_List ['', 'easeIn', 'easeOut', 'easeInOut', 'random']*/;
 
+MoveMode2_List
+    :   '匀速移动'|'慢-快'|'快-慢'|'慢-快-慢'|'快-慢-快'
+    /*MoveMode2_List ['', 'in', 'out', 'in-out', 'center']*/;
+
 NameMap_List
     :   '确定'|'取消'|'操作失败'|'光标移动'|'打开界面'|'读档'|'存档'|'获得道具'|'回血'|'宝石'|'炸弹'|'飞行器'|'开关门'|'上下楼'|'跳跃'|'破墙镐'|'破冰镐'|'阻激夹域'|'穿脱装备'|'商店'
     /*NameMap_List ['确定','取消','操作失败','光标移动','打开界面','读档','存档','获得道具','回血','宝石','炸弹','飞行器','开关门','上下楼','跳跃','破墙镐','破冰镐','阻激夹域','穿脱装备','商店']*/;
diff --git a/_server/MotaActionParser.js b/_server/MotaActionParser.js
index 06c2081..9c9fd92 100644
--- a/_server/MotaActionParser.js
+++ b/_server/MotaActionParser.js
@@ -542,6 +542,12 @@ MotaActionParser = function () {
           );
         }
         break;
+      case "cgtextList":
+        this.next = MotaActionBlocks["cgtextList_s"].xmlText([
+          data.textList,
+          this.next,
+        ]);
+        break;
       case "cgtext": // cg对话框
         var buildcgDrawing = function (obj) {
           obj = MotaActionFunctions.processcgDrawing(obj || []);
@@ -560,16 +566,12 @@ MotaActionParser = function () {
           return res;
         };
         this.next = MotaActionBlocks["cgtext_s"].xmlText([
-          data.bg,
-          data.memory,
           data.WindowSkin,
+          data.index,
           data.head.name,
-          data.name || "",
           data.head.px || -300,
           data.time,
           data.wait,
-          data.sound,
-          data.text,
           buildcgDrawing(data.bodyList),
           this.next,
         ]);
@@ -620,6 +622,20 @@ MotaActionParser = function () {
       case "setcgs":
         this.next = MotaActionBlocks["setcgs_s"].xmlText([data.img, this.next]);
         break;
+      case "setmusics":
+        this.next = MotaActionBlocks["setmusics_s"].xmlText([
+          data.bgm,
+          this.next,
+        ]);
+        break;
+      case "introAndLoop":
+        this.next = MotaActionBlocks["introAndLoop_s"].xmlText([
+          data.intro,
+          data.time,
+          data.loop,
+          this.next,
+        ]);
+        break;
       case "comment": // 注释
         this.next = MotaActionBlocks["comment_s"].xmlText([
           this.EvalString_Multi(data.text),
@@ -713,141 +729,7 @@ MotaActionParser = function () {
           this.next,
         ]);
         break;
-      case "setanimate":
-        var buildanimateDrawableimage = function (obj) {
-          obj = MotaActionFunctions.processanimateDrawableimage(obj || []);
-          var res = null;
-          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
-            res = MotaActionBlocks["animateDrawableimage"].xmlText([
-              one[0],
-              one[1],
-              one[2],
-              one[3],
-              one[4],
-              one[5],
-              one[6],
-              one[7],
-              one[8],
-              one[9],
-              one[10],
-              one[11],
-              one[12],
-              one[13],
-              one[14],
-              one[15],
-              one[16],
-              one[17],
-              one[18],
-              one[19],
-              one[20],
-              one[21],
-              one[22],
-              res,
-            ]);
-          }
-          return res;
-        };
-        var buildanimateDrawablesound = function (obj) {
-          obj = MotaActionFunctions.processanimateDrawablesound(obj || []);
-          var res = null;
-          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
-            res = MotaActionBlocks["animateDrawablesound"].xmlText([
-              one[0],
-              one[1],
-              one[2],
-              res,
-            ]);
-          }
-          return res;
-        };
-        this.next = MotaActionBlocks["setanimate_s"].xmlText([
-          data.name,
-          data.px,
-          data.py,
-          data.width,
-          data.height,
-          data.allFarme,
-          buildanimateDrawableimage(data.imageList),
-          buildanimateDrawablesound(data.soundList),
-          this.next,
-        ]);
-        break;
-      case "clearanimate":
-        this.next = MotaActionBlocks["clearanimate_s"].xmlText([this.next]);
-        break;
-      case "deleteanimate":
-        this.next = MotaActionBlocks["deleteanimate_s"].xmlText([
-          data.name,
-          this.next,
-        ]);
-        break;
-      case "playanimate":
-        this.next = MotaActionBlocks["playanimate_s"].xmlText([
-          data.name,
-          data.x,
-          data.y,
-          data.hero,
-          data.scalex,
-          data.scaley,
-          this.next,
-        ]);
-        break;
-      case "animationDrawable":
-        var buildanimationDrawableimage = function (obj) {
-          obj = MotaActionFunctions.processanimationDrawableimage(obj || []);
-          var res = null;
-          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
-            res = MotaActionBlocks["animationDrawableimage"].xmlText([
-              one[0],
-              one[1],
-              one[2],
-              one[3],
-              one[4],
-              one[5],
-              one[6],
-              one[7],
-              one[8],
-              one[9],
-              one[10],
-              one[11],
-              one[12],
-              one[13],
-              one[14],
-              one[15],
-              one[16],
-              one[17],
-              one[18],
-              one[19],
-              one[20],
-              res,
-            ]);
-          }
-          return res;
-        };
-        var buildanimationDrawablesound = function (obj) {
-          obj = MotaActionFunctions.processanimationDrawablesound(obj || []);
-          var res = null;
-          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
-            res = MotaActionBlocks["animationDrawablesound"].xmlText([
-              one[0],
-              one[1],
-              one[2],
-              res,
-            ]);
-          }
-          return res;
-        };
-        this.next = MotaActionBlocks["animationDrawable_s"].xmlText([
-          data.allFarme,
-          data.color,
-          "rgba(" + data.color + ")",
-          data.globalAlpha,
-          buildanimationDrawableimage(data.imageList),
-          buildanimationDrawablesound(data.soundList),
-          this.next,
-        ]);
-        break;
-
+     
       case "tip":
         this.next = MotaActionBlocks["tip_s"].xmlText([
           data.text,
@@ -1135,6 +1017,64 @@ MotaActionParser = function () {
           ]);
         }
         break;
+        case "animationDrawable":
+        var buildanimationDrawableimage = function (obj) {
+          obj = MotaActionFunctions.processanimationDrawableimage(obj || []);
+          var res = null;
+          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
+            res = MotaActionBlocks["animationDrawableimage"].xmlText([
+              one[0],
+              one[1],
+              one[2],
+              one[3],
+              one[4],
+              one[5],
+              one[6],
+              one[7],
+              one[8],
+              one[9],
+              one[10],
+              one[11],
+              one[12],
+              one[13],
+              one[14],
+              one[15],
+              one[16],
+              one[17],
+              one[18],
+              one[19],
+              one[20],
+              one[21],
+              one[22],
+              res,
+            ]);
+          }
+          return res;
+        };
+        var buildanimationDrawablesound = function (obj) {
+          obj = MotaActionFunctions.processanimationDrawablesound(obj || []);
+          var res = null;
+          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
+            res = MotaActionBlocks["animationDrawablesound"].xmlText([
+              one[0],
+              one[1],
+              one[2],
+              res,
+            ]);
+          }
+          return res;
+        };
+        this.next = MotaActionBlocks["animationDrawable_s"].xmlText([
+          data.allFarme,
+          data.color,
+          "rgba(" + data.color + ")",
+          data.globalAlpha,
+          buildanimationDrawableimage(data.imageList),
+          buildanimationDrawablesound(data.soundList),
+          this.next,
+        ]);
+        break;
+
       case "jumpHero": // 跳跃勇士
         if (data.dxy) {
           this.next = MotaActionBlocks["jumpHero_1_s"].xmlText([
@@ -1219,12 +1159,161 @@ MotaActionParser = function () {
           ]);
         }
         break;
+        case "animateResize": // 停止所有动画
+        this.next = MotaActionBlocks["animateResize_s"].xmlText([
+          data.name,
+          data.id||"",
+          data.centerX||"",
+          data.centerY||"",
+          data.hero||false,
+          data.reverse||false,
+          data.loop||false,
+          data.async || false,
+          this.next,
+        ]);
+        break;
+        case "pauseAnimate": // 暂停所有动画
+        this.next = MotaActionBlocks["pauseAnimate_s"].xmlText([
+          data.id||"",
+          this.next,
+        ]);
+        break;
+        case "remuseAnimate": // 继续所有动画
+        this.next = MotaActionBlocks["remuseAnimate_s"].xmlText([
+          data.id||"",
+          this.next,
+        ]);
+        break;
       case "stopAnimate": // 停止所有动画
         this.next = MotaActionBlocks["stopAnimate_s"].xmlText([
+          data.id||"",
           data.doCallback || false,
           this.next,
         ]);
         break;
+        case "setanimate":
+        var buildanimateDrawableimage = function (obj) {
+          obj = MotaActionFunctions.processanimateDrawableimage(obj || []);
+          var res = null;
+          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
+            res = MotaActionBlocks["animateDrawableimage"].xmlText([
+              one[0],
+              one[1],
+              one[2],
+              one[3],
+              one[4],
+              one[5],
+              one[6],
+              one[7],
+              one[8],
+              one[9],
+              one[10],
+              one[11],
+              one[12],
+              one[13],
+              one[14],
+              one[15],
+              one[16],
+              one[17],
+              one[18],
+              one[19],
+              one[20],
+              one[21],
+              one[22],
+              res,
+            ]);
+          }
+          return res;
+        };
+        var buildanimateDrawablesound = function (obj) {
+          obj = MotaActionFunctions.processanimateDrawablesound(obj || []);
+          var res = null;
+          for (var ii = obj.length - 1, one; (one = obj[ii]); ii--) {
+            res = MotaActionBlocks["animateDrawablesound"].xmlText([
+              one[0],
+              one[1],
+              one[2],
+              res,
+            ]);
+          }
+          return res;
+        };
+        this.next = MotaActionBlocks["setanimate_s"].xmlText([
+          data.name,
+          data.px,
+          data.py,
+          data.width,
+          data.height,
+          data.allFarme,
+          buildanimateDrawableimage(data.imageList),
+          buildanimateDrawablesound(data.soundList),
+          this.next,
+        ]);
+        break;
+      case "clearanimate":
+        this.next = MotaActionBlocks["clearanimate_s"].xmlText([
+          data.id||"",
+          this.next]);
+        break;
+        case "animateloop":
+        this.next = MotaActionBlocks["animateloop_s"].xmlText([
+          data.id||"",
+          data.loop,
+          this.next]);
+        break;
+        case "animatemove":
+        this.next = MotaActionBlocks["animatemove_s"].xmlText([
+          data.id||0,
+          data.px||0,
+          data.py||0,
+          data.relative,
+          data.time||0,
+          data.style||"",
+          this.next]);
+        break;
+        case "moveAnimate":
+          this.next = MotaActionBlocks["moveAnimate_s"].xmlText([
+            data.id||0,
+            data.px||0,
+            data.py||0,
+            data.time||0,
+            data.relative,
+            data.style||"",
+            this.next]);
+          break;
+       
+        case "animatereverse":
+        this.next = MotaActionBlocks["animatereverse_s"].xmlText([
+          data.id||"",
+          data.reverse,
+          this.next]);
+        break;
+        case "animatepause":
+        this.next = MotaActionBlocks["animatepause_s"].xmlText([
+          data.id||"",
+          data.pause,
+          this.next]);
+        break;
+      case "deleteanimate":
+        this.next = MotaActionBlocks["deleteanimate_s"].xmlText([
+          data.name,
+          this.next,
+        ]);
+        break;
+      case "playanimate":
+        this.next = MotaActionBlocks["playanimate_s"].xmlText([
+          data.name,
+          data.id||"",
+          data.x,
+          data.y,
+          data.hero,
+          data.scalex||1,
+          data.scaley||1,
+          data.loop||false,
+          data.reverse||false,
+          this.next,
+        ]);
+        break;
       case "setViewport": // 设置视角
         if (data.dxy) {
           this.next = MotaActionBlocks["setViewport_1_s"].xmlText([
@@ -1405,7 +1494,6 @@ MotaActionParser = function () {
         this.next = MotaActionBlocks["setWeather_s"].xmlText([
           data.name,
           data.level || 1,
-          data.keep || false,
           this.next,
         ]);
         break;
@@ -1506,27 +1594,6 @@ MotaActionParser = function () {
           ]);
         }
         break;
-      case "playStereo":
-        this.next = MotaActionBlocks["playStereo_s"].xmlText([
-          data.name,
-          data.left,
-          data.right,
-          data.split || true,
-          this.next,
-        ]);
-        break;
-      case "moveStereo":
-        this.next = MotaActionBlocks["moveStereo_s"].xmlText([
-          data.name,
-          data.left,
-          data.right,
-          data.split || true,
-          data.leftTo,
-          data.rightTo,
-          data.time,
-          this.next,
-        ]);
-        break;
       case "playSound":
         var knownItems = MotaActionBlocks["NameMap_List"].options.map(function (
           one
@@ -1996,7 +2063,7 @@ MotaActionParser = function () {
           this.next,
         ]);
         break;
-      case "changebg": // 等待多少毫秒
+      case "changebg": // 改变背景
         this.next = MotaActionBlocks["changebg_s"].xmlText([
           data.img1 ?? "",
           data.memory1 ?? false,
@@ -2007,6 +2074,18 @@ MotaActionParser = function () {
           this.next,
         ]);
         break;
+        case "drawbackground": // 绘制背景
+        this.next = MotaActionBlocks["drawbackground_s"].xmlText([
+          data.image ?? "",
+          data.memory ?? false,
+          this.next,
+        ]);
+        break;
+        case "clearbackground": // 清除背景
+        this.next = MotaActionBlocks["clearbackground_s"].xmlText([
+          this.next,
+        ]);
+        break;
       case "wait": // 等待用户操作
         var case_waitList = null;
         if (data.data) {
@@ -2107,6 +2186,7 @@ MotaActionParser = function () {
         this.next = MotaActionBlocks["drawWarning_s"].xmlText([
           data.x,
           data.y,
+          data.large,
           data.size,
           data?.text,
           data?.text2,
@@ -2916,56 +2996,7 @@ MotaActionParser = function () {
     });
     return list;
   };
-  MotaActionFunctions.processovertext = function (overList) {
-    var list = [];
-    overList.forEach(function (one) {
-      list.push([
-        one.text,
-        one.sound,
-        one.time,
-        one.textColor,
-        one.boldColor,
-        one.font,
-      ]);
-    });
-    return list;
-  };
-  MotaActionFunctions.processanimationDrawableimage = function (overList) {
-    var list = [];
-    overList.forEach(function (one) {
-      list.push([
-        one.image,
-        one.beforefarme,
-        one.globalAlpha,
-        one.cx,
-        one.cy,
-        one.cw,
-        one.ch,
-        one.x,
-        one.y,
-        one.w,
-        one.h,
-        one.afterfarme,
-        one.aglobalAlpha,
-        one.acx,
-        one.acy,
-        one.acw,
-        one.ach,
-        one.ax,
-        one.ay,
-        one.aw,
-        one.ah,
-      ]);
-    });
-    return list;
-  };
-  MotaActionFunctions.processanimationDrawablesound = function (overList) {
-    var list = [];
-    overList.forEach(function (one) {
-      list.push([one.sound, one.startfarme, one.stopbefore]);
-    });
-    return list;
-  };
+  
   MotaActionFunctions.processanimateDrawableimage = function (overList) {
     var list = [];
     overList.forEach(function (one) {
@@ -3004,6 +3035,21 @@ MotaActionParser = function () {
     });
     return list;
   };
+  MotaActionFunctions.processovertext = function (overList) {
+    var list = [];
+    overList.forEach(function (one) {
+      list.push([
+        one.text,
+        one.sound,
+        one.time,
+        one.textColor,
+        one.boldColor,
+        one.font,
+      ]);
+    });
+    return list;
+  };
+ 
   MotaActionFunctions.processMultiLoc = function (EvalString_0, EvalString_1) {
     var floorstr = "";
     if (EvalString_0 && EvalString_1) {
@@ -3277,6 +3323,45 @@ MotaActionParser = function () {
     return str;
   };
 
+  MotaActionFunctions.processanimationDrawableimage = function (overList) {
+    var list = [];
+    overList.forEach(function (one) {
+      list.push([
+        one.image,
+        one.beforefarme,
+        one.globalAlpha,
+        one.cx,
+        one.cy,
+        one.cw,
+        one.ch,
+        one.x,
+        one.y,
+        one.w,
+        one.h,
+        one.angle,
+        one.afterfarme,
+        one.aglobalAlpha,
+        one.acx,
+        one.acy,
+        one.acw,
+        one.ach,
+        one.ax,
+        one.ay,
+        one.aw,
+        one.ah,
+        one.aangle
+      ]);
+    });
+    return list;
+  };
+  MotaActionFunctions.processanimationDrawablesound = function (overList) {
+    var list = [];
+    overList.forEach(function (one) {
+      list.push([one.sound, one.startfarme, one.stopbefore]);
+    });
+    return list;
+  };
+
   MotaActionFunctions.replaceFromName = function (str) {
     if (!str || MotaActionFunctions.disableReplace) return str;
     var map = {},
diff --git a/_server/config.json b/_server/config.json
deleted file mode 100644
index bdcb574..0000000
--- a/_server/config.json
+++ /dev/null
@@ -1 +0,0 @@
-{"viewportLoc":[0,0],"editorLastFloorId":"jiedao"}
\ No newline at end of file
diff --git a/_server/editor_blockly.js b/_server/editor_blockly.js
index 4f0f62d..a3433b1 100644
--- a/_server/editor_blockly.js
+++ b/_server/editor_blockly.js
@@ -354,6 +354,21 @@ editor_blockly = function () {
                         editor.uievent.previewUI([{ type: "drawImage", image: obj.name, x: obj.loc[0], y: obj.loc[1] }]);
                     }
                     break;
+                    case 'animateResize_s': // 显示动画
+                    if (obj.name && obj.centerX&&obj.centerY) {
+                        editor.uievent.previewUI([{ type: "fillRect", x:obj.centerX-5, y: obj.centerY-5, width:10, height: 10, style:"#FF0000"}]);
+                    }
+                    break;
+                    case 'playanimate_s': // 显示帧动画
+                    if (obj.name && obj.x&&obj.y) {
+                        editor.uievent.previewUI([{ type: "fillRect", x:obj.x-5, y: obj.y-5, width:10, height: 10, style:"#FF0000"}]);
+                    }
+                    break;
+                    case 'animatemove_s': // 移动帧动画
+                   
+                        editor.uievent.previewUI([{ type: "fillRect", x:obj.px-5, y: obj.py-5, width:10, height: 10, style:"#FF0000"}]);
+                    
+                    break;
                 case 'setCurtain_0_s': // 更改色调
                     if (obj.color) {
                         editor.uievent.previewUI([{ type: "fillRect", x: 0, y: 0, width: core.__PIXELS__, height: core.__PIXELS__, style: obj.color }]);
diff --git a/_server/editor_blocklyconfig.js b/_server/editor_blocklyconfig.js
index 6c99618..b0f038c 100644
--- a/_server/editor_blocklyconfig.js
+++ b/_server/editor_blocklyconfig.js
@@ -132,8 +132,8 @@ editor_blocklyconfig = function () {
         MotaActionFunctions.actionParser.parse(
           {
             time: 160,
-            openSound: "door.mp3",
-            closeSound: "door.mp3",
+            openSound: "door.opus",
+            closeSound: "door.opus",
             keys: { yellowKey: 1, orangeKey: 1 },
           },
           "doorInfo"
@@ -142,9 +142,9 @@ editor_blocklyconfig = function () {
         MotaActionBlocks["mainStyle_m"].xmlText(),
         MotaActionFunctions.actionParser.parse(
           {
-            背景音乐: "bgm.mp3",
-            确定: "confirm.mp3",
-            攻击: "attack.mp3",
+            背景音乐: "bgm.opus",
+            确定: "confirm.opus",
+            攻击: "attack.opus",
             背景图: "bg.jpg",
             领域: "zone",
             文件名: "file.jpg",
diff --git a/_server/table/comment.js b/_server/table/comment.js
index f8660c0..98f7d1a 100644
--- a/_server/table/comment.js
+++ b/_server/table/comment.js
@@ -116,6 +116,14 @@ var comment_c456ea59_6018_45ef_8bcc_211a24c627dc = {
 					"_lint": true,
 					"_docs": "能否使用或装备",
 					"_data": "当前能否使用或装备该道具,仅对cls不为items有效。null表示始终不可使用但可装备"
+				},
+				"canBatchUse": {
+					"_leaf": true,
+					"_type": "textarea",
+					"_string": true,
+					"_lint": true,
+					"_docs": "能否批量使用",
+					"_data": "该道具是否允许批量使用,仅对cls为tools或constants有效,true表示可批量使用。建议同时勾选【回放不绘制道具栏】"
 				}
 			}
 		},
@@ -192,7 +200,14 @@ var comment_c456ea59_6018_45ef_8bcc_211a24c627dc = {
 				"mdef": {
 					"_leaf": true,
 					"_type": "textarea",
-					"_data": "法抗(写为小数)"
+					"_docs": "法抗",
+					"_data": "百分数,如10为10%法抗,怪物受到勇士的法术伤害减免比例"
+
+				},
+				"spell": {
+					"_leaf": true,
+					"_type": "textarea",
+					"_data": "法强"
 				},
 				"speed": {
 					"_leaf": true,
@@ -389,7 +404,7 @@ var comment_c456ea59_6018_45ef_8bcc_211a24c627dc = {
 				}
 			}
 		},
-		"enemys_template": { 'name': '新敌人', 'hp': 0, 'atk': 0, 'def': 0, 'money': 0, 'exp': 0, 'point': 0, 'special': [] },
+		"enemys_template": { 'name': '新敌人', 'hp': 0, 'atk': 0, 'def': 0, 'mdef': 0, 'speed': 1, 'money': 0, 'exp': 0, 'point': 0, 'special': [] },
 
 
 		// --------------------------- 【图块属性】相关的表格配置 --------------------------- //
diff --git a/_server/table/data.comment.js b/_server/table/data.comment.js
index 3a76bd7..17faa7f 100644
--- a/_server/table/data.comment.js
+++ b/_server/table/data.comment.js
@@ -264,10 +264,27 @@ var data_comment_c456ea59_6018_45ef_8bcc_211a24c627dc = {
 							"_type": "textarea",
 							"_data": "防御"
 						},
+						"spell": {
+							"_leaf": true,
+							"_type": "textarea",
+							"_data": "法强",
+						},
 						"mdef": {
 							"_leaf": true,
 							"_type": "textarea",
-							"_data": "法强"
+							"_data": "法抗百分比",
+						},
+						"matk": {
+							"_leaf": true,
+							"_type": "textarea",
+							"_data": "100为法强转化为等值魔攻",
+							"_docs": "魔攻比例"
+						},
+						"mhp": {
+							"_leaf": true,
+							"_type": "textarea",
+							"_data": "100为法强转化为等值护盾",
+							"_docs": "护盾比例"
 						},
 						"speed": {
 							"_leaf": true,
@@ -435,6 +452,12 @@ var data_comment_c456ea59_6018_45ef_8bcc_211a24c627dc = {
 					"_type": "textarea",
 					"_docs": "绿宝石值",
 				},
+				"yellowGem": {
+					"_leaf": true,
+					"_type": "textarea",
+					"_docs": "黄宝石值",
+				},
+
 				"redPotion": {
 					"_leaf": true,
 					"_type": "textarea",
diff --git a/_server/table/plugins.comment.js b/_server/table/plugins.comment.js
index a02e298..a9cb7a1 100644
--- a/_server/table/plugins.comment.js
+++ b/_server/table/plugins.comment.js
@@ -122,6 +122,18 @@ var plugins_comment_c456ea59_6018_45ef_8bcc_211a24c627dc = {
 				"_type": "textarea",
 				"_range": "typeof(thiseval)=='string' || thiseval==null",
 				"_data": "音乐鉴赏"
+			},
+			"intro&loop": {
+				"_leaf": true,
+				"_type": "textarea",
+				"_range": "typeof(thiseval)=='string' || thiseval==null",
+				"_data": "背景音乐拼接"
+			},
+			"scrollingText": {
+				"_leaf": true,
+				"_type": "textarea",
+				"_range": "typeof(thiseval)=='string' || thiseval==null",
+				"_data": "接收&发送在线留言"
 			}
 		}
 		if (obj[key]) return obj[key];
diff --git a/index.html b/index.html
index c49e005..a316bc3 100644
--- a/index.html
+++ b/index.html
@@ -47,6 +47,20 @@
           startImageDiv.classList.add("startImageDivAnimation");
           // 注释下面这句话以禁止单击立刻跳过开场动画
           startImageBackgroundDiv.onclick = onAnimationEnd;
+          // 新增加载图片轮播逻辑
+    const slides = document.querySelectorAll('.loading-slide');
+    let currentSlide = 0;
+    
+    const slideInterval = setInterval(() => {
+        slides[currentSlide].classList.remove('active');
+        currentSlide = (currentSlide + 1) % slides.length;
+        slides[currentSlide].classList.add('active');
+    }, 3000);
+
+    // 在加载完成时清除定时器
+    window.addEventListener('load', () => {
+        clearInterval(slideInterval);
+    });
         };
         startImageLogo.onerror = function () {};
         startImageLogo.src = "logo.png";
@@ -58,6 +72,14 @@
       <img id="musicBtn" />
       <div id="startPanel">
         <div id="startTop">
+           <!-- 新增图片展示区域 -->
+    <div id='startTopGallery'>
+       <img src='project/images/bg_1511.webp' class='loading-slide active' />
+      <img src='project/images/bg_1521.webp' class='loading-slide'/>
+      <img src='project/images/bg_2011.webp' class='loading-slide'/>
+     <img src='project/images/bg_2521.webp' class='loading-slide'/>
+      <img src='project/images/bg_3026.webp' class='loading-slide'/>
+  </div>
           <div id="startTopProgressBar">
             <div id="startTopProgress"></div>
           </div>
diff --git a/libs/control.js b/libs/control.js
index c6e4f30..93b4a0a 100644
--- a/libs/control.js
+++ b/libs/control.js
@@ -1,4 +1,3 @@
-
 /*
 control.js:游戏主要逻辑控制
 主要负责status相关内容,以及各种变量获取/存储
@@ -7,57 +6,93 @@ control.js:游戏主要逻辑控制
 
 "use strict";
 
-function control () {
-    this._init();
+function control() {
+  this._init();
 }
 
 control.prototype._init = function () {
-    this.controldata = functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a.control;
-    this.renderFrameFuncs = [];
-    this.replayActions = [];
-    this.weathers = {};
-    this.resizes = [];
-    this.noAutoEvents = true;
-    this.updateNextFrame = false;
-    // --- 注册系统的animationFrame
-    this.registerAnimationFrame("totalTime", false, this._animationFrame_totalTime);
-    this.registerAnimationFrame("autoSave", true, this._animationFrame_autoSave);
-    this.registerAnimationFrame("globalAnimate", true, this._animationFrame_globalAnimate);
-    this.registerAnimationFrame("animate", true, this._animationFrame_animate);
-    this.registerAnimationFrame("heroMoving", true, this._animationFrame_heroMoving);
-    this.registerAnimationFrame("weather", true, this._animationFrame_weather);
-    this.registerAnimationFrame("tip", true, this._animateFrame_tip);
-    this.registerAnimationFrame("parallelDo", false, this._animationFrame_parallelDo);
-    // --- 注册系统的天气
-    this.registerWeather("rain", this._weather_rain, this._animationFrame_weather_rain);
-    this.registerWeather("snow", this._weather_snow, this._animationFrame_weather_snow);
-    this.registerWeather("fog", this._weather_fog, this.__animateFrame_weather_image);
-    this.registerWeather("cloud", this._weather_cloud, this.__animateFrame_weather_image);
-    this.registerWeather("sun", this._weather_sun, this._animationFrame_weather_sun);
-    // --- 注册系统的replay
-    this.registerReplayAction("move", this._replayAction_move);
-    this.registerReplayAction("item", this._replayAction_item);
-    this.registerReplayAction("equip", this._replayAction_equip);
-    this.registerReplayAction("unEquip", this._replayAction_unEquip);
-    this.registerReplayAction("saveEquip", this._replayAction_saveEquip);
-    this.registerReplayAction("loadEquip", this._replayAction_loadEquip);
-    this.registerReplayAction("fly", this._replayAction_fly);
-    this.registerReplayAction("shop", this._replayAction_shop);
-    this.registerReplayAction("turn", this._replayAction_turn);
-    this.registerReplayAction("getNext", this._replayAction_getNext);
-    this.registerReplayAction("moveDirectly", this._replayAction_moveDirectly);
-    this.registerReplayAction("key", this._replayAction_key);
-    this.registerReplayAction("click", this._replayAction_click);
-    this.registerReplayAction("ignoreInput", this._replayAction_ignoreInput);
-    this.registerReplayAction("no", this._replayAction_no);
-    // --- 注册系统的resize
-    this.registerResize("gameGroup", this._resize_gameGroup);
-    this.registerResize("canvas", this._resize_canvas);
-    this.registerResize("statusBar", this._resize_statusBar);
-    this.registerResize("status", this._resize_status);
-    this.registerResize("toolBar", this._resize_toolBar);
-    this.registerResize("tools", this._resize_tools);
-}
+  this.controldata = functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a.control;
+  this.renderFrameFuncs = [];
+  this.replayActions = [];
+  this.weathers = {};
+  this.resizes = [];
+  this.noAutoEvents = true;
+  this.updateNextFrame = false;
+  // --- 注册系统的animationFrame
+  this.registerAnimationFrame(
+    "totalTime",
+    false,
+    this._animationFrame_totalTime
+  );
+  this.registerAnimationFrame("autoSave", true, this._animationFrame_autoSave);
+  this.registerAnimationFrame(
+    "globalAnimate",
+    true,
+    this._animationFrame_globalAnimate
+  );
+  this.registerAnimationFrame("animate", true, this._animationFrame_animate);
+  this.registerAnimationFrame(
+    "heroMoving",
+    true,
+    this._animationFrame_heroMoving
+  );
+  this.registerAnimationFrame("weather", true, this._animationFrame_weather);
+  this.registerAnimationFrame("tip", true, this._animateFrame_tip);
+  this.registerAnimationFrame(
+    "parallelDo",
+    false,
+    this._animationFrame_parallelDo
+  );
+  // --- 注册系统的天气
+  this.registerWeather(
+    "rain",
+    this._weather_rain,
+    this._animationFrame_weather_rain
+  );
+  this.registerWeather(
+    "snow",
+    this._weather_snow,
+    this._animationFrame_weather_snow
+  );
+  this.registerWeather(
+    "fog",
+    this._weather_fog,
+    this.__animateFrame_weather_image
+  );
+  this.registerWeather(
+    "cloud",
+    this._weather_cloud,
+    this.__animateFrame_weather_image
+  );
+  this.registerWeather(
+    "sun",
+    this._weather_sun,
+    this._animationFrame_weather_sun
+  );
+  // --- 注册系统的replay
+  this.registerReplayAction("move", this._replayAction_move);
+  this.registerReplayAction("item", this._replayAction_item);
+  this.registerReplayAction("equip", this._replayAction_equip);
+  this.registerReplayAction("unEquip", this._replayAction_unEquip);
+  this.registerReplayAction("saveEquip", this._replayAction_saveEquip);
+  this.registerReplayAction("loadEquip", this._replayAction_loadEquip);
+  this.registerReplayAction("fly", this._replayAction_fly);
+  this.registerReplayAction("shop", this._replayAction_shop);
+  this.registerReplayAction("turn", this._replayAction_turn);
+  this.registerReplayAction("getNext", this._replayAction_getNext);
+  this.registerReplayAction("moveDirectly", this._replayAction_moveDirectly);
+  this.registerReplayAction("key", this._replayAction_key);
+  this.registerReplayAction("click", this._replayAction_click);
+  this.registerReplayAction("ignoreInput", this._replayAction_ignoreInput);
+  this.registerReplayAction("no", this._replayAction_no);
+  // --- 注册系统的resize
+  this.registerResize("gameGroup", this._resize_gameGroup);
+  this.registerResize("canvas", this._resize_canvas);
+  this.registerResize("statusBar", this._resize_statusBar);
+  this.registerResize("status", this._resize_status);
+  this.registerResize("toolBar", this._resize_toolBar);
+  this.registerResize("tools", this._resize_tools);
+};
 
 // ------ requestAnimationFrame 相关 ------ //
 
@@ -65,1542 +100,1938 @@ control.prototype._init = function () {
 // name:名称,可用来作为注销使用;needPlaying:是否只在游戏运行时才执行(在标题界面不执行)
 // func:要执行的函数,或插件中的函数名;可接受timestamp(从页面加载完毕到当前所经过的时间)作为参数
 control.prototype.registerAnimationFrame = function (name, needPlaying, func) {
-    this.unregisterAnimationFrame(name);
-    this.renderFrameFuncs.push({ name: name, needPlaying: needPlaying, func: func });
-}
+  this.unregisterAnimationFrame(name);
+  this.renderFrameFuncs.push({
+    name: name,
+    needPlaying: needPlaying,
+    func: func,
+  });
+};
 
 ////// 注销一个 animationFrame //////
 control.prototype.unregisterAnimationFrame = function (name) {
-    this.renderFrameFuncs = this.renderFrameFuncs.filter(function (x) { return x.name != name; });
-}
+  this.renderFrameFuncs = this.renderFrameFuncs.filter(function (x) {
+    return x.name != name;
+  });
+};
 
 ////// 设置requestAnimationFrame //////
 control.prototype._setRequestAnimationFrame = function () {
-    this._checkRequestAnimationFrame();
-    core.animateFrame.totalTime = Math.max(core.animateFrame.totalTime, core.getLocalStorage('totalTime', 0));
-    var loop = function (timestamp) {
-        core.control.renderFrameFuncs.forEach(function (b) {
-            if (b.func) {
-                try {
-                    if (core.isPlaying() || !b.needPlaying)
-                        core.doFunc(b.func, core.control, timestamp);
-                }
-                catch (e) {
-                    console.error(e);
-                    console.error("ERROR in requestAnimationFrame[" + b.name + "]:已自动注销该项。");
-                    core.unregisterAnimationFrame(b.name);
-                }
-            }
-        })
-        window.requestAnimationFrame(loop);
-    }
+  this._checkRequestAnimationFrame();
+  core.animateFrame.totalTime = Math.max(
+    core.animateFrame.totalTime,
+    core.getLocalStorage("totalTime", 0)
+  );
+  var loop = function (timestamp) {
+    core.control.renderFrameFuncs.forEach(function (b) {
+      if (b.func) {
+        try {
+          if (core.isPlaying() || !b.needPlaying)
+            core.doFunc(b.func, core.control, timestamp);
+        } catch (e) {
+          console.error(e);
+          console.error(
+            "ERROR in requestAnimationFrame[" + b.name + "]:已自动注销该项。"
+          );
+          core.unregisterAnimationFrame(b.name);
+        }
+      }
+    });
     window.requestAnimationFrame(loop);
-}
+  };
+  window.requestAnimationFrame(loop);
+};
 
 control.prototype._checkRequestAnimationFrame = function () {
-    (function () {
-        var lastTime = 0;
-        var vendors = ['webkit', 'moz'];
-        for (var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
-            window.requestAnimationFrame = window[vendors[x] + 'RequestAnimationFrame'];
-            window.cancelAnimationFrame = window[vendors[x] + 'CancelAnimationFrame'] ||    // Webkit中此取消方法的名字变了
-                window[vendors[x] + 'CancelRequestAnimationFrame'];
-        }
+  (function () {
+    var lastTime = 0;
+    var vendors = ["webkit", "moz"];
+    for (var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
+      window.requestAnimationFrame =
+        window[vendors[x] + "RequestAnimationFrame"];
+      window.cancelAnimationFrame =
+        window[vendors[x] + "CancelAnimationFrame"] || // Webkit中此取消方法的名字变了
+        window[vendors[x] + "CancelRequestAnimationFrame"];
+    }
 
-        if (!window.requestAnimationFrame) {
-            window.requestAnimationFrame = function (callback, element) {
-                var currTime = new Date().getTime();
-                var timeToCall = Math.max(0, 16.7 - (currTime - lastTime));
-                var id = window.setTimeout(function () {
-                    callback(currTime + timeToCall);
-                }, timeToCall);
-                lastTime = currTime + timeToCall;
-                return id;
-            };
-        }
-        if (!window.cancelAnimationFrame) {
-            window.cancelAnimationFrame = function (id) {
-                clearTimeout(id);
-            };
-        }
-    }());
-}
+    if (!window.requestAnimationFrame) {
+      window.requestAnimationFrame = function (callback, element) {
+        var currTime = new Date().getTime();
+        var timeToCall = Math.max(0, 16.7 - (currTime - lastTime));
+        var id = window.setTimeout(function () {
+          callback(currTime + timeToCall);
+        }, timeToCall);
+        lastTime = currTime + timeToCall;
+        return id;
+      };
+    }
+    if (!window.cancelAnimationFrame) {
+      window.cancelAnimationFrame = function (id) {
+        clearTimeout(id);
+      };
+    }
+  })();
+};
 
 control.prototype._animationFrame_totalTime = function (timestamp) {
-    core.animateFrame.totalTime += timestamp - core.animateFrame.totalTimeStart;
-    core.animateFrame.totalTimeStart = timestamp;
-    if (core.isPlaying()) {
-        core.status.hero.statistics.totalTime = core.animateFrame.totalTime;
-        core.status.hero.statistics.currTime += timestamp - (core.status.hero.statistics.start || timestamp);
-        core.status.hero.statistics.start = timestamp;
-    }
-}
+  core.animateFrame.totalTime += timestamp - core.animateFrame.totalTimeStart;
+  core.animateFrame.totalTimeStart = timestamp;
+  if (core.isPlaying()) {
+    core.status.hero.statistics.totalTime = core.animateFrame.totalTime;
+    core.status.hero.statistics.currTime +=
+      timestamp - (core.status.hero.statistics.start || timestamp);
+    core.status.hero.statistics.start = timestamp;
+  }
+};
 
 control.prototype._animationFrame_autoSave = function (timestamp) {
-    if (timestamp - core.saves.autosave.time <= 5000) return;
-    core.control.checkAutosave();
-    core.saves.autosave.time = timestamp;
-}
+  if (timestamp - core.saves.autosave.time <= 5000) return;
+  core.control.checkAutosave();
+  core.saves.autosave.time = timestamp;
+};
 
 control.prototype._animationFrame_globalAnimate = function (timestamp) {
-    if (timestamp - core.animateFrame.globalTime <= core.values.animateSpeed) return;
-    core.status.globalAnimateStatus++;
-    if (core.status.floorId) {
-        // Global Animate
-        core.status.globalAnimateObjs.forEach(function (block) {
-            core.drawBlock(block, core.status.globalAnimateStatus);
-        });
+  if (timestamp - core.animateFrame.globalTime <= core.values.animateSpeed)
+    return;
+  core.status.globalAnimateStatus++;
+  if (core.status.floorId) {
+    // Global Animate
+    core.status.globalAnimateObjs.forEach(function (block) {
+      core.drawBlock(block, core.status.globalAnimateStatus);
+    });
 
-        // Global floor images
-        core.maps._drawFloorImages(core.status.floorId, core.canvas.bg, 'bg', core.status.floorAnimateObjs || [], core.status.globalAnimateStatus);
-        core.maps._drawFloorImages(core.status.floorId, core.canvas.fg, 'fg', core.status.floorAnimateObjs || [], core.status.globalAnimateStatus);
+    // Global floor images
+    core.maps._drawFloorImages(
+      core.status.floorId,
+      core.canvas.bg,
+      "bg",
+      core.status.floorAnimateObjs || [],
+      core.status.globalAnimateStatus
+    );
+    core.maps._drawFloorImages(
+      core.status.floorId,
+      core.canvas.fg,
+      "fg",
+      core.status.floorAnimateObjs || [],
+      core.status.globalAnimateStatus
+    );
 
-        // Global Autotile Animate
-        core.status.autotileAnimateObjs.forEach(function (block) {
-            core.maps._drawAutotileAnimate(block, core.status.globalAnimateStatus);
-        });
+    // Global Autotile Animate
+    core.status.autotileAnimateObjs.forEach(function (block) {
+      core.maps._drawAutotileAnimate(block, core.status.globalAnimateStatus);
+    });
 
-        // Global hero animate
-        if ((core.status.hero || {}).animate && core.status.heroMoving == 0 && main.mode == 'play' && !core.status.preview.enabled) {
-            core.drawHero('stop', null, core.status.globalAnimateStatus);
-        }
+    // Global hero animate
+    if (
+      (core.status.hero || {}).animate &&
+      core.status.heroMoving == 0 &&
+      main.mode == "play" &&
+      !core.status.preview.enabled
+    ) {
+      core.drawHero("stop", null, core.status.globalAnimateStatus);
     }
-    // Box animate
-    core.drawBoxAnimate();
-    core.animateFrame.globalTime = timestamp;
-}
+  }
+  // Box animate
+  core.drawBoxAnimate();
+  core.animateFrame.globalTime = timestamp;
+};
 
 control.prototype._animationFrame_animate = function (timestamp) {
-    if (timestamp - core.animateFrame.animateTime < 50 || !core.status.animateObjs || core.status.animateObjs.length == 0) return;
-    core.clearMap('animate');
-    // 更新帧
-    for (var i = 0; i < core.status.animateObjs.length; i++) {
-        var obj = core.status.animateObjs[i];
-        if (obj.index == obj.animate.frames.length) {
-            (function (callback) {
-                setTimeout(function () {
-                    if (callback) callback();
-                });
-            })(obj.callback);
-        }
+  if (
+    timestamp - core.animateFrame.animateTime < 50 ||
+    !core.status.animateObjs ||
+    core.status.animateObjs.length == 0
+  )
+    return;
+  core.clearMap("animate");
+  // 更新帧
+  for (var i = 0; i < core.status.animateObjs.length; i++) {
+    var obj = core.status.animateObjs[i];
+    if (obj.index == obj.animate.frames.length) {
+      (function (callback) {
+        setTimeout(function () {
+          if (callback) callback();
+        });
+      })(obj.callback);
     }
-    core.status.animateObjs = core.status.animateObjs.filter(function (obj) {
-        return obj.index < obj.animate.frames.length;
-    });
-    core.status.animateObjs.forEach(function (obj) {
-        if (obj.hero) {
-            core.maps._drawAnimateFrame('animate', obj.animate, core.status.heroCenter.px, core.status.heroCenter.py, obj.index++);
-        } else {
-            core.maps._drawAnimateFrame('animate', obj.animate, obj.centerX, obj.centerY, obj.index++);
-        }
-    });
-    core.animateFrame.animateTime = timestamp;
-}
+  }
+  core.status.animateObjs = core.status.animateObjs.filter(function (obj) {
+    return obj.index < obj.animate.frames.length;
+  });
+  core.status.animateObjs.forEach(function (obj) {
+    if (obj.hero) {
+      core.maps._drawAnimateFrame(
+        "animate",
+        obj.animate,
+        core.status.heroCenter.px,
+        core.status.heroCenter.py,
+        obj.index++
+      );
+    } else {
+      core.maps._drawAnimateFrame(
+        "animate",
+        obj.animate,
+        obj.centerX,
+        obj.centerY,
+        obj.index++
+      );
+    }
+  });
+  core.animateFrame.animateTime = timestamp;
+};
 
 control.prototype._animationFrame_heroMoving = function (timestamp) {
-    if (core.status.heroMoving <= 0) return;
-    // 换腿
-    if (timestamp - core.animateFrame.moveTime > core.values.moveSpeed) {
-        core.animateFrame.leftLeg = !core.animateFrame.leftLeg;
-        core.animateFrame.moveTime = timestamp;
-    }
-    core.drawHero(core.animateFrame.leftLeg ? 'leftFoot' : 'rightFoot', 4 * core.status.heroMoving);
-}
+  if (core.status.heroMoving <= 0) return;
+  // 换腿
+  if (timestamp - core.animateFrame.moveTime > core.values.moveSpeed) {
+    core.animateFrame.leftLeg = !core.animateFrame.leftLeg;
+    core.animateFrame.moveTime = timestamp;
+  }
+  core.drawHero(
+    core.animateFrame.leftLeg ? "leftFoot" : "rightFoot",
+    4 * core.status.heroMoving
+  );
+};
 
 control.prototype._animationFrame_weather = function (timestamp) {
-    var weather = core.animateFrame.weather, type = weather.type;
-    if (!core.dymCanvas.weather || !core.control.weathers[type] || !core.control.weathers[type].frameFunc) return;
-    try {
-        core.doFunc(core.control.weathers[type].frameFunc, core.control, timestamp, core.animateFrame.weather.level);
-    } catch (e) {
-        console.error(e);
-        console.error("ERROR in weather[" + type + "]:已自动注销该项。");
-        core.unregisterWeather(type);
-    }
-}
+  var weather = core.animateFrame.weather,
+    type = weather.type;
+  if (
+    !core.dymCanvas.weather ||
+    !core.control.weathers[type] ||
+    !core.control.weathers[type].frameFunc
+  )
+    return;
+  try {
+    core.doFunc(
+      core.control.weathers[type].frameFunc,
+      core.control,
+      timestamp,
+      core.animateFrame.weather.level
+    );
+  } catch (e) {
+    console.error(e);
+    console.error("ERROR in weather[" + type + "]:已自动注销该项。");
+    core.unregisterWeather(type);
+  }
+};
 
 control.prototype._animationFrame_weather_rain = function (timestamp, level) {
-    if (timestamp - core.animateFrame.weather.time < 30) return;
-    var ctx = core.dymCanvas.weather, ox = core.bigmap.offsetX, oy = core.bigmap.offsetY;
-    core.clearMap('weather');
-    ctx.strokeStyle = 'rgba(174,194,224,0.8)';
-    ctx.lineWidth = 1;
-    ctx.lineCap = 'round';
+  if (timestamp - core.animateFrame.weather.time < 30) return;
+  var ctx = core.dymCanvas.weather,
+    ox = core.bigmap.offsetX,
+    oy = core.bigmap.offsetY;
+  core.clearMap("weather");
+  ctx.strokeStyle = "rgba(174,194,224,0.8)";
+  ctx.lineWidth = 1;
+  ctx.lineCap = "round";
 
-    core.animateFrame.weather.nodes.forEach(function (p) {
-        ctx.beginPath();
-        ctx.moveTo(p.x - ox, p.y - oy);
-        ctx.lineTo(p.x + p.l * p.xs - ox, p.y + p.l * p.ys - oy);
-        ctx.stroke();
+  core.animateFrame.weather.nodes.forEach(function (p) {
+    ctx.beginPath();
+    ctx.moveTo(p.x - ox, p.y - oy);
+    ctx.lineTo(p.x + p.l * p.xs - ox, p.y + p.l * p.ys - oy);
+    ctx.stroke();
 
-        p.x += p.xs;
-        p.y += p.ys;
-        if (p.x > core.bigmap.width * 32 || p.y > core.bigmap.height * 32) {
-            p.x = Math.random() * core.bigmap.width * 32;
-            p.y = -10;
-        }
+    p.x += p.xs;
+    p.y += p.ys;
+    if (p.x > core.bigmap.width * 32 || p.y > core.bigmap.height * 32) {
+      p.x = Math.random() * core.bigmap.width * 32;
+      p.y = -10;
+    }
+  });
 
-    });
-
-    ctx.fill();
-    core.animateFrame.weather.time = timestamp;
-}
+  ctx.fill();
+  core.animateFrame.weather.time = timestamp;
+};
 
 control.prototype._animationFrame_weather_snow = function (timestamp, level) {
-    if (timestamp - core.animateFrame.weather.time < 30) return;
-    var ctx = core.dymCanvas.weather, ox = core.bigmap.offsetX, oy = core.bigmap.offsetY;
-    core.clearMap('weather');
-    ctx.fillStyle = "rgba(255, 255, 255, 0.8)";
-    ctx.beginPath();
-    core.animateFrame.weather.data = core.animateFrame.weather.data || 0;
-    core.animateFrame.weather.data += 0.01;
+  if (timestamp - core.animateFrame.weather.time < 30) return;
+  var ctx = core.dymCanvas.weather,
+    ox = core.bigmap.offsetX,
+    oy = core.bigmap.offsetY;
+  core.clearMap("weather");
+  ctx.fillStyle = "rgba(255, 255, 255, 0.8)";
+  ctx.beginPath();
+  core.animateFrame.weather.data = core.animateFrame.weather.data || 0;
+  core.animateFrame.weather.data += 0.01;
 
-    var angle = core.animateFrame.weather.data;
-    core.animateFrame.weather.nodes.forEach(function (p) {
-        ctx.moveTo(p.x - ox, p.y - oy);
-        ctx.arc(p.x - ox, p.y - oy, p.r, 0, Math.PI * 2, true);
-        // update
-        p.x += Math.sin(angle) * core.animateFrame.weather.level;
-        p.y += Math.cos(angle + p.d) + 1 + p.r / 2;
-        if (p.x > core.bigmap.width * 32 + 5 || p.x < -5 || p.y > core.bigmap.height * 32) {
-            if (Math.random() > 1 / 3) {
-                p.x = Math.random() * core.bigmap.width * 32;
-                p.y = -10;
-            }
-            else {
-                if (Math.sin(angle) > 0)
-                    p.x = -5;
-                else
-                    p.x = core.bigmap.width * 32 + 5;
-                p.y = Math.random() * core.bigmap.height * 32;
-            }
-        }
-    });
-    ctx.fill();
-    core.animateFrame.weather.time = timestamp;
-}
+  var angle = core.animateFrame.weather.data;
+  core.animateFrame.weather.nodes.forEach(function (p) {
+    ctx.moveTo(p.x - ox, p.y - oy);
+    ctx.arc(p.x - ox, p.y - oy, p.r, 0, Math.PI * 2, true);
+    // update
+    p.x += Math.sin(angle) * core.animateFrame.weather.level;
+    p.y += Math.cos(angle + p.d) + 1 + p.r / 2;
+    if (
+      p.x > core.bigmap.width * 32 + 5 ||
+      p.x < -5 ||
+      p.y > core.bigmap.height * 32
+    ) {
+      if (Math.random() > 1 / 3) {
+        p.x = Math.random() * core.bigmap.width * 32;
+        p.y = -10;
+      } else {
+        if (Math.sin(angle) > 0) p.x = -5;
+        else p.x = core.bigmap.width * 32 + 5;
+        p.y = Math.random() * core.bigmap.height * 32;
+      }
+    }
+  });
+  ctx.fill();
+  core.animateFrame.weather.time = timestamp;
+};
 
 control.prototype.__animateFrame_weather_image = function (timestamp, level) {
-    if (timestamp - core.animateFrame.weather.time < 30) return;
-    var node = core.animateFrame.weather.nodes[0];
-    var image = node.image;
-    if (!image) return;
-    core.clearMap('weather');
-    core.setAlpha('weather', node.level / 500);
-    var wind = 1.5;
-    var width = image.width, height = image.height;
-    node.x += node.dx * wind;
-    node.y += (2 * node.dy - 1) * wind;
-    if (node.x + 3 * width <= core._PX_) {
-        node.x += 4 * width;
-        while (node.x > 0) node.x -= width;
+  if (timestamp - core.animateFrame.weather.time < 30) return;
+  var node = core.animateFrame.weather.nodes[0];
+  var image = node.image;
+  if (!image) return;
+  core.clearMap("weather");
+  core.setAlpha("weather", node.level / 500);
+  var wind = 1.5;
+  var width = image.width,
+    height = image.height;
+  node.x += node.dx * wind;
+  node.y += (2 * node.dy - 1) * wind;
+  if (node.x + 3 * width <= core._PX_) {
+    node.x += 4 * width;
+    while (node.x > 0) node.x -= width;
+  }
+  node.dy += node.delta;
+  if (node.dy >= 1) {
+    node.delta = -0.001;
+  } else if (node.dy <= 0) {
+    node.delta = 0.001;
+  }
+  if (node.y + 3 * height <= core._PY_) {
+    node.y += 4 * height;
+    while (node.y > 0) node.y -= height;
+  } else if (node.y >= 0) {
+    node.y -= height;
+  }
+  for (var i = 0; i < 3; ++i) {
+    for (var j = 0; j < 3; ++j) {
+      if (
+        node.x + (i + 1) * width <= 0 ||
+        node.x + i * width >= core._PX_ ||
+        node.y + (j + 1) * height <= 0 ||
+        node.y + j * height >= core._PY_
+      )
+        continue;
+      core.drawImage("weather", image, node.x + i * width, node.y + j * height);
     }
-    node.dy += node.delta;
-    if (node.dy >= 1) {
-        node.delta = -0.001;
-    } else if (node.dy <= 0) {
-        node.delta = 0.001;
-    }
-    if (node.y + 3 * height <= core._PY_) {
-        node.y += 4 * height;
-        while (node.y > 0) node.y -= height;
-    }
-    else if (node.y >= 0) {
-        node.y -= height;
-    }
-    for (var i = 0; i < 3; ++i) {
-        for (var j = 0; j < 3; ++j) {
-            if (node.x + (i + 1) * width <= 0 || node.x + i * width >= core._PX_
-                || node.y + (j + 1) * height <= 0 || node.y + j * height >= core._PY_)
-                continue;
-            core.drawImage('weather', image, node.x + i * width, node.y + j * height);
-        }
-    }
-    core.setAlpha('weather', 1);
-    core.animateFrame.weather.time = timestamp;
-}
+  }
+  core.setAlpha("weather", 1);
+  core.animateFrame.weather.time = timestamp;
+};
 
 control.prototype._animationFrame_weather_sun = function (timestamp, level) {
-    if (timestamp - core.animateFrame.weather.time < 30) return;
-    var node = core.animateFrame.weather.nodes[0];
-    var opacity = node.opacity + node.delta;
-    if (opacity > level / 10 + 0.3 || opacity < level / 10 - 0.3)
-        node.delta = -node.delta;
-    node.opacity = opacity;
-    core.setOpacity('weather', core.clamp(opacity, 0, 1));
-    core.animateFrame.weather.time = timestamp;
-}
+  if (timestamp - core.animateFrame.weather.time < 30) return;
+  var node = core.animateFrame.weather.nodes[0];
+  var opacity = node.opacity + node.delta;
+  if (opacity > level / 10 + 0.3 || opacity < level / 10 - 0.3)
+    node.delta = -node.delta;
+  node.opacity = opacity;
+  core.setOpacity("weather", core.clamp(opacity, 0, 1));
+  core.animateFrame.weather.time = timestamp;
+};
 
 control.prototype._animateFrame_tip = function (timestamp) {
-    if (core.animateFrame.tip == null) return;
-    var tip = core.animateFrame.tip;
-    if (timestamp - tip.time <= 30) return;
-    var delta = timestamp - tip.time;
-    tip.time = timestamp;
+  if (core.animateFrame.tip == null) return;
+  var tip = core.animateFrame.tip;
+  if (timestamp - tip.time <= 30) return;
+  var delta = timestamp - tip.time;
+  tip.time = timestamp;
 
-    core.setFont('data', "16px Arial");
-    core.setTextAlign('data', 'left');
-    core.clearMap('data', 0, 0, core._PX_, 50);
-    core.ui._drawTip_drawOne(tip);
-    if (tip.stage == 1) {
-        tip.opacity += 0.05;
-        if (tip.opacity >= 0.6) {
-            tip.stage = 2;
-            tip.displayTime = 0;
-        }
-    } else if (tip.stage == 2) {
-        tip.displayTime += delta;
-        if (tip.displayTime >= 1000) tip.stage = 3;
-    } else tip.opacity -= 0.05;
-
-    if (tip.opacity <= 0) {
-        core.animateFrame.tip = null;
+  core.setFont("data", "16px Arial");
+  core.setTextAlign("data", "left");
+  core.clearMap("data", 0, 0, core._PX_, 50);
+  core.ui._drawTip_drawOne(tip);
+  if (tip.stage == 1) {
+    tip.opacity += 0.05;
+    if (tip.opacity >= 0.6) {
+      tip.stage = 2;
+      tip.displayTime = 0;
     }
-}
+  } else if (tip.stage == 2) {
+    tip.displayTime += delta;
+    if (tip.displayTime >= 1000) tip.stage = 3;
+  } else tip.opacity -= 0.05;
+
+  if (tip.opacity <= 0) {
+    core.animateFrame.tip = null;
+  }
+};
 
 control.prototype._animationFrame_parallelDo = function (timestamp) {
-    core.control.controldata.parallelDo(timestamp);
-}
+  core.control.controldata.parallelDo(timestamp);
+};
 
 // ------ 标题界面的处理 ------ //
 
 ////// 显示游戏开始界面 //////
 control.prototype.showStartAnimate = function (noAnimate, callback) {
-    this._showStartAnimate_resetDom();
-    if (core.flags.startUsingCanvas || noAnimate)
-        return this._showStartAnimate_finished(core.flags.startUsingCanvas, callback);
-    core.hideWithAnimate(core.dom.startTop, 20, function () {
-        core.control._showStartAnimate_finished(false, callback);
-    });
-}
+  this._showStartAnimate_resetDom();
+  if (core.flags.startUsingCanvas || noAnimate)
+    return this._showStartAnimate_finished(
+      core.flags.startUsingCanvas,
+      callback
+    );
+  core.hideWithAnimate(core.dom.startTop, 20, function () {
+    core.control._showStartAnimate_finished(false, callback);
+  });
+};
 
 control.prototype._showStartAnimate_resetDom = function () {
-    core.dom.startPanel.style.opacity = 1;
-    core.dom.startPanel.style.display = "block";
-    core.dom.startTop.style.opacity = 1;
-    core.dom.startTop.style.display = "block";
-    core.dom.startButtonGroup.style.display = 'none';
-    core.dom.startButtons.style.display = 'block';
-    core.dom.levelChooseButtons.style.display = 'none';
-    core.status.played = false;
-    core.clearStatus();
-    core.clearMap('all');
-    core.dom.musicBtn.style.display = 'block';
-    core.setMusicBtn();
-    // 重置音量
-    core.events.setVolume(1, 0);
-    core.updateStatusBar();
-}
+  core.dom.startPanel.style.opacity = 1;
+  core.dom.startPanel.style.display = "block";
+  core.dom.startTop.style.opacity = 1;
+  core.dom.startTop.style.display = "block";
+  core.dom.startButtonGroup.style.display = "none";
+  core.dom.startButtons.style.display = "block";
+  core.dom.levelChooseButtons.style.display = "none";
+  core.status.played = false;
+  core.clearStatus();
+  core.clearMap("all");
+  core.dom.musicBtn.style.display = "block";
+  core.setMusicBtn();
+  // 重置音量
+  core.events.setVolume(1, 0);
+  core.updateStatusBar();
+};
 
 control.prototype._showStartAnimate_finished = function (start, callback) {
-    core.dom.startTop.style.display = 'none';
-    core.dom.startButtonGroup.style.display = 'block';
-    main.selectedButton = null;
-    main.selectButton(0);
-    if (start) core.startGame();
-    if (callback) callback();
-}
+  core.dom.startTop.style.display = "none";
+  core.dom.startButtonGroup.style.display = "block";
+  main.selectedButton = null;
+  main.selectButton(0);
+  if (start) core.startGame();
+  if (callback) callback();
+};
 
 ////// 隐藏游戏开始界面 //////
 control.prototype.hideStartAnimate = function (callback) {
-    core.hideWithAnimate(core.dom.startPanel, 20, callback);
-}
+  core.hideWithAnimate(core.dom.startPanel, 20, callback);
+};
 
 ////// 游戏是否已经开始 //////
 control.prototype.isPlaying = function () {
-    return core.status.played;
-}
+  return core.status.played;
+};
 
 ////// 清除游戏状态和数据 //////
 control.prototype.clearStatus = function () {
-    // 停止各个Timeout和Interval
-    for (var i in core.timeout) {
-        clearTimeout(core.timeout[i]);
-        core.timeout[i] = null;
-    }
-    for (var i in core.interval) {
-        clearInterval(core.interval[i]);
-        core.interval[i] = null;
-    }
-    core.status = {};
-    core.clearStatusBar();
-    core.deleteAllCanvas();
-    core.status.played = false;
-}
+  // 停止各个Timeout和Interval
+  for (var i in core.timeout) {
+    clearTimeout(core.timeout[i]);
+    core.timeout[i] = null;
+  }
+  for (var i in core.interval) {
+    clearInterval(core.interval[i]);
+    core.interval[i] = null;
+  }
+  core.status = {};
+  core.clearStatusBar();
+  core.deleteAllCanvas();
+  core.status.played = false;
+};
 
 control.prototype._initStatistics = function (totalTime) {
-    if (!core.isset(core.status.hero.statistics))
-        core.status.hero.statistics = {
-            'totalTime': totalTime,
-            'currTime': 0,
-            'hp': 0,
-            "battle": 0,
-            'money': 0,
-            'exp': 0,
-            'battleDamage': 0,
-            'poisonDamage': 0,
-            'extraDamage': 0,
-            'moveDirectly': 0,
-            'ignoreSteps': 0,
-        }
-}
+  if (!core.isset(core.status.hero.statistics))
+    core.status.hero.statistics = {
+      totalTime: totalTime,
+      currTime: 0,
+      hp: 0,
+      battle: 0,
+      money: 0,
+      exp: 0,
+      battleDamage: 0,
+      poisonDamage: 0,
+      extraDamage: 0,
+      moveDirectly: 0,
+      ignoreSteps: 0,
+    };
+};
 
 // ------ 自动寻路,人物行走 ------ //
 
 ////// 清除自动寻路路线 //////
 control.prototype.clearAutomaticRouteNode = function (x, y) {
-    core.clearMap('route', x * 32 + 5 - core.status.automaticRoute.offsetX, y * 32 + 5 - core.status.automaticRoute.offsetY, 27, 27);
-}
+  core.clearMap(
+    "route",
+    x * 32 + 5 - core.status.automaticRoute.offsetX,
+    y * 32 + 5 - core.status.automaticRoute.offsetY,
+    27,
+    27
+  );
+};
 
 ////// 停止自动寻路操作 //////
 control.prototype.stopAutomaticRoute = function () {
-    if (!core.status.played) return;
-    core.status.automaticRoute.autoHeroMove = false;
-    core.status.automaticRoute.autoStep = 0;
-    core.status.automaticRoute.destStep = 0;
-    core.status.automaticRoute.movedStep = 0;
-    core.status.automaticRoute.autoStepRoutes = [];
-    core.status.automaticRoute.destX = null;
-    core.status.automaticRoute.destY = null;
-    core.status.automaticRoute.lastDirection = null;
-    core.status.heroStop = true;
-    if (core.status.automaticRoute.moveStepBeforeStop.length == 0)
-        core.deleteCanvas('route');
-}
+  if (!core.status.played) return;
+  core.status.automaticRoute.autoHeroMove = false;
+  core.status.automaticRoute.autoStep = 0;
+  core.status.automaticRoute.destStep = 0;
+  core.status.automaticRoute.movedStep = 0;
+  core.status.automaticRoute.autoStepRoutes = [];
+  core.status.automaticRoute.destX = null;
+  core.status.automaticRoute.destY = null;
+  core.status.automaticRoute.lastDirection = null;
+  core.status.heroStop = true;
+  if (core.status.automaticRoute.moveStepBeforeStop.length == 0)
+    core.deleteCanvas("route");
+};
 
 ////// 保存剩下的寻路,并停止 //////
 control.prototype.saveAndStopAutomaticRoute = function () {
-    var automaticRoute = core.status.automaticRoute;
-    if (automaticRoute.moveStepBeforeStop.length == 0) {
-        automaticRoute.moveStepBeforeStop = automaticRoute.autoStepRoutes.slice(automaticRoute.autoStep - 1);
-        if (automaticRoute.moveStepBeforeStop.length >= 1)
-            automaticRoute.moveStepBeforeStop[0].step -= automaticRoute.movedStep;
-    }
-    this.stopAutomaticRoute();
-}
+  var automaticRoute = core.status.automaticRoute;
+  if (automaticRoute.moveStepBeforeStop.length == 0) {
+    automaticRoute.moveStepBeforeStop = automaticRoute.autoStepRoutes.slice(
+      automaticRoute.autoStep - 1
+    );
+    if (automaticRoute.moveStepBeforeStop.length >= 1)
+      automaticRoute.moveStepBeforeStop[0].step -= automaticRoute.movedStep;
+  }
+  this.stopAutomaticRoute();
+};
 
 ////// 继续剩下的自动寻路操作 //////
 control.prototype.continueAutomaticRoute = function () {
-    // 此函数只应由events.afterOpenDoor和events.afterBattle调用
-    var moveStep = core.status.automaticRoute.moveStepBeforeStop;
-    //core.status.automaticRoute.moveStepBeforeStop = [];
-    if (moveStep.length === 0 || (moveStep.length === 1 && moveStep[0].step === 1)) {
-        core.status.automaticRoute.moveStepBeforeStop = [];
-    }
-    else {
-        core.setAutoHeroMove(moveStep);
-    }
-}
+  // 此函数只应由events.afterOpenDoor和events.afterBattle调用
+  var moveStep = core.status.automaticRoute.moveStepBeforeStop;
+  //core.status.automaticRoute.moveStepBeforeStop = [];
+  if (
+    moveStep.length === 0 ||
+    (moveStep.length === 1 && moveStep[0].step === 1)
+  ) {
+    core.status.automaticRoute.moveStepBeforeStop = [];
+  } else {
+    core.setAutoHeroMove(moveStep);
+  }
+};
 
 ////// 清空剩下的自动寻路列表 //////
 control.prototype.clearContinueAutomaticRoute = function (callback) {
-    core.deleteCanvas('route');
-    core.status.automaticRoute.moveStepBeforeStop = [];
-    if (callback) callback();
-}
+  core.deleteCanvas("route");
+  core.status.automaticRoute.moveStepBeforeStop = [];
+  if (callback) callback();
+};
 
 ////// 设置自动寻路路线 //////
 control.prototype.setAutomaticRoute = function (destX, destY, stepPostfix) {
-    if (!core.status.played || core.status.lockControl) return;
-    if (this._setAutomaticRoute_isMoving(destX, destY)) return;
-    if (this._setAutomaticRoute_isTurning(destX, destY, stepPostfix)) return;
-    if (this._setAutomaticRoute_clickMoveDirectly(destX, destY, stepPostfix)) return;
-    // 找寻自动寻路路线
-    var moveStep = core.automaticRoute(destX, destY);
-    if (moveStep.length == 0 && (destX != core.status.hero.loc.x || destY != core.status.hero.loc.y || stepPostfix.length == 0))
-        return;
-    moveStep = moveStep.concat(stepPostfix);
-    core.status.automaticRoute.destX = destX;
-    core.status.automaticRoute.destY = destY;
-    this._setAutomaticRoute_drawRoute(moveStep);
-    this._setAutomaticRoute_setAutoSteps(moveStep);
-    // 立刻移动
-    core.setAutoHeroMove();
-}
+  if (!core.status.played || core.status.lockControl) return;
+  if (this._setAutomaticRoute_isMoving(destX, destY)) return;
+  if (this._setAutomaticRoute_isTurning(destX, destY, stepPostfix)) return;
+  if (this._setAutomaticRoute_clickMoveDirectly(destX, destY, stepPostfix))
+    return;
+  // 找寻自动寻路路线
+  var moveStep = core.automaticRoute(destX, destY);
+  if (
+    moveStep.length == 0 &&
+    (destX != core.status.hero.loc.x ||
+      destY != core.status.hero.loc.y ||
+      stepPostfix.length == 0)
+  )
+    return;
+  moveStep = moveStep.concat(stepPostfix);
+  core.status.automaticRoute.destX = destX;
+  core.status.automaticRoute.destY = destY;
+  this._setAutomaticRoute_drawRoute(moveStep);
+  this._setAutomaticRoute_setAutoSteps(moveStep);
+  // 立刻移动
+  core.setAutoHeroMove();
+};
 
 control.prototype._setAutomaticRoute_isMoving = function (destX, destY) {
-    if (core.status.automaticRoute.autoHeroMove) {
-        var lastX = core.status.automaticRoute.destX, lastY = core.status.automaticRoute.destY;
-        core.stopAutomaticRoute();
-        // 双击瞬移
-        if (lastX == destX && lastY == destY) {
-            core.status.automaticRoute.moveDirectly = true;
-            setTimeout(function () {
-                if (core.status.automaticRoute.moveDirectly && core.status.heroMoving == 0) {
-                    core.control.tryMoveDirectly(destX, destY);
-                }
-                core.status.automaticRoute.moveDirectly = false;
-            }, core.values.moveSpeed);
+  if (core.status.automaticRoute.autoHeroMove) {
+    var lastX = core.status.automaticRoute.destX,
+      lastY = core.status.automaticRoute.destY;
+    core.stopAutomaticRoute();
+    // 双击瞬移
+    if (lastX == destX && lastY == destY) {
+      core.status.automaticRoute.moveDirectly = true;
+      setTimeout(function () {
+        if (
+          core.status.automaticRoute.moveDirectly &&
+          core.status.heroMoving == 0
+        ) {
+          core.control.tryMoveDirectly(destX, destY);
         }
-        return true;
+        core.status.automaticRoute.moveDirectly = false;
+      }, core.values.moveSpeed);
     }
-    return false;
-}
+    return true;
+  }
+  return false;
+};
 
-control.prototype._setAutomaticRoute_isTurning = function (destX, destY, stepPostfix) {
-    if (destX == core.status.hero.loc.x && destY == core.status.hero.loc.y && stepPostfix.length == 0) {
-        if (core.timeout.turnHeroTimeout == null) {
-            var routeLength = core.status.route.length;
-            core.timeout.turnHeroTimeout = setTimeout(function () {
-                if (core.status.route.length == routeLength) core.turnHero();
-                clearTimeout(core.timeout.turnHeroTimeout);
-                core.timeout.turnHeroTimeout = null;
-            }, 250);
-        }
-        else {
-            clearTimeout(core.timeout.turnHeroTimeout);
-            core.timeout.turnHeroTimeout = null;
-            core.getNextItem();
-        }
-        return true;
+control.prototype._setAutomaticRoute_isTurning = function (
+  destX,
+  destY,
+  stepPostfix
+) {
+  if (
+    destX == core.status.hero.loc.x &&
+    destY == core.status.hero.loc.y &&
+    stepPostfix.length == 0
+  ) {
+    if (core.timeout.turnHeroTimeout == null) {
+      var routeLength = core.status.route.length;
+      core.timeout.turnHeroTimeout = setTimeout(function () {
+        if (core.status.route.length == routeLength) core.turnHero();
+        clearTimeout(core.timeout.turnHeroTimeout);
+        core.timeout.turnHeroTimeout = null;
+      }, 250);
+    } else {
+      clearTimeout(core.timeout.turnHeroTimeout);
+      core.timeout.turnHeroTimeout = null;
+      core.getNextItem();
     }
-    if (core.timeout.turnHeroTimeout != null) return true;
-    return false;
-}
+    return true;
+  }
+  if (core.timeout.turnHeroTimeout != null) return true;
+  return false;
+};
 
-control.prototype._setAutomaticRoute_clickMoveDirectly = function (destX, destY, stepPostfix) {
-    // 单击瞬间移动
-    if (core.status.heroStop && core.status.heroMoving == 0) {
-        if (stepPostfix.length <= 1 && !core.hasFlag('__noClickMove__') && core.control.tryMoveDirectly(destX, destY))
-            return true;
-    }
-    return false;
-}
+control.prototype._setAutomaticRoute_clickMoveDirectly = function (
+  destX,
+  destY,
+  stepPostfix
+) {
+  // 单击瞬间移动
+  if (core.status.heroStop && core.status.heroMoving == 0) {
+    if (
+      stepPostfix.length <= 1 &&
+      !core.hasFlag("__noClickMove__") &&
+      core.control.tryMoveDirectly(destX, destY)
+    )
+      return true;
+  }
+  return false;
+};
 
 control.prototype._setAutomaticRoute_drawRoute = function (moveStep) {
-    // 计算绘制区域的宽高,并尽可能小的创建route层
-    var sx = core.bigmap.width * 32, sy = core.bigmap.height * 32, dx = 0, dy = 0;
-    moveStep.forEach(function (t) {
-        sx = Math.min(sx, t.x * 32); dx = Math.max(dx, t.x * 32);
-        sy = Math.min(sy, t.y * 32); dy = Math.max(dy, t.y * 32);
-    });
-    core.status.automaticRoute.offsetX = sx;
-    core.status.automaticRoute.offsetY = sy;
-    var ctx = core.createCanvas('route', sx - core.bigmap.offsetX, sy - core.bigmap.offsetY, dx - sx + 32, dy - sy + 32, 95);
-    ctx.fillStyle = '#bfbfbf';
-    ctx.strokeStyle = '#bfbfbf';
-    ctx.lineWidth = 8;
-    for (var m = 0; m < moveStep.length; m++) {
-        if (m == moveStep.length - 1) {
-            ctx.fillRect(moveStep[m].x * 32 + 10 - sx, moveStep[m].y * 32 + 10 - sy, 12, 12);
-        }
-        else {
-            ctx.beginPath();
-            var cx = moveStep[m].x * 32 + 16 - sx, cy = moveStep[m].y * 32 + 16 - sy;
-            var currDir = moveStep[m].direction, nextDir = moveStep[m + 1].direction;
-            ctx.moveTo(cx - core.utils.scan[currDir].x * 11, cy - core.utils.scan[currDir].y * 11);
-            ctx.lineTo(cx, cy);
-            ctx.lineTo(cx + core.utils.scan[nextDir].x * 11, cy + core.utils.scan[nextDir].y * 11);
-            ctx.stroke();
-        }
+  // 计算绘制区域的宽高,并尽可能小的创建route层
+  var sx = core.bigmap.width * 32,
+    sy = core.bigmap.height * 32,
+    dx = 0,
+    dy = 0;
+  moveStep.forEach(function (t) {
+    sx = Math.min(sx, t.x * 32);
+    dx = Math.max(dx, t.x * 32);
+    sy = Math.min(sy, t.y * 32);
+    dy = Math.max(dy, t.y * 32);
+  });
+  core.status.automaticRoute.offsetX = sx;
+  core.status.automaticRoute.offsetY = sy;
+  var ctx = core.createCanvas(
+    "route",
+    sx - core.bigmap.offsetX,
+    sy - core.bigmap.offsetY,
+    dx - sx + 32,
+    dy - sy + 32,
+    95
+  );
+  ctx.fillStyle = "#bfbfbf";
+  ctx.strokeStyle = "#bfbfbf";
+  ctx.lineWidth = 8;
+  for (var m = 0; m < moveStep.length; m++) {
+    if (m == moveStep.length - 1) {
+      ctx.fillRect(
+        moveStep[m].x * 32 + 10 - sx,
+        moveStep[m].y * 32 + 10 - sy,
+        12,
+        12
+      );
+    } else {
+      ctx.beginPath();
+      var cx = moveStep[m].x * 32 + 16 - sx,
+        cy = moveStep[m].y * 32 + 16 - sy;
+      var currDir = moveStep[m].direction,
+        nextDir = moveStep[m + 1].direction;
+      ctx.moveTo(
+        cx - core.utils.scan[currDir].x * 11,
+        cy - core.utils.scan[currDir].y * 11
+      );
+      ctx.lineTo(cx, cy);
+      ctx.lineTo(
+        cx + core.utils.scan[nextDir].x * 11,
+        cy + core.utils.scan[nextDir].y * 11
+      );
+      ctx.stroke();
     }
-}
+  }
+};
 
 control.prototype._setAutomaticRoute_setAutoSteps = function (moveStep) {
-    // 路线转autoStepRoutes
-    var step = 0, currStep = null;
-    moveStep.forEach(function (t) {
-        var dir = t.direction;
-        if (currStep == null || currStep == dir)
-            step++;
-        else {
-            core.status.automaticRoute.autoStepRoutes.push({ 'direction': currStep, 'step': step });
-            step = 1;
-        }
-        currStep = dir;
-    });
-    core.status.automaticRoute.autoStepRoutes.push({ 'direction': currStep, 'step': step });
-}
+  // 路线转autoStepRoutes
+  var step = 0,
+    currStep = null;
+  moveStep.forEach(function (t) {
+    var dir = t.direction;
+    if (currStep == null || currStep == dir) step++;
+    else {
+      core.status.automaticRoute.autoStepRoutes.push({
+        direction: currStep,
+        step: step,
+      });
+      step = 1;
+    }
+    currStep = dir;
+  });
+  core.status.automaticRoute.autoStepRoutes.push({
+    direction: currStep,
+    step: step,
+  });
+};
 
 ////// 设置勇士的自动行走路线 //////
 control.prototype.setAutoHeroMove = function (steps) {
-    steps = steps || core.status.automaticRoute.autoStepRoutes;
-    if (steps.length == 0) return;
-    core.status.automaticRoute.autoStepRoutes = steps;
-    core.status.automaticRoute.autoHeroMove = true;
-    core.status.automaticRoute.autoStep = 1;
-    core.status.automaticRoute.destStep = steps[0].step;
-    core.moveHero(steps[0].direction);
-}
+  steps = steps || core.status.automaticRoute.autoStepRoutes;
+  if (steps.length == 0) return;
+  core.status.automaticRoute.autoStepRoutes = steps;
+  core.status.automaticRoute.autoHeroMove = true;
+  core.status.automaticRoute.autoStep = 1;
+  core.status.automaticRoute.destStep = steps[0].step;
+  core.moveHero(steps[0].direction);
+};
 
 ////// 设置行走的效果动画 //////
 control.prototype.setHeroMoveInterval = function (callback) {
-    if (core.status.heroMoving > 0) return;
-    if (core.status.replay.speed == 24) {
-        if (callback) callback();
-        return;
+  if (core.status.heroMoving > 0) return;
+  if (core.status.replay.speed == 24) {
+    if (callback) callback();
+    return;
+  }
+
+  core.status.heroMoving = 1;
+
+  var toAdd = 1;
+  if (core.status.replay.speed > 3) toAdd = 2;
+  if (core.status.replay.speed > 6) toAdd = 4;
+  if (core.status.replay.speed > 12) toAdd = 8;
+
+  core.interval.heroMoveInterval = window.setInterval(function () {
+    core.status.heroMoving += toAdd;
+    if (core.status.heroMoving >= 8) {
+      clearInterval(core.interval.heroMoveInterval);
+      core.status.heroMoving = 0;
+      if (callback) callback();
     }
-
-    core.status.heroMoving = 1;
-
-    var toAdd = 1;
-    if (core.status.replay.speed > 3) toAdd = 2;
-    if (core.status.replay.speed > 6) toAdd = 4;
-    if (core.status.replay.speed > 12) toAdd = 8;
-
-    core.interval.heroMoveInterval = window.setInterval(function () {
-        core.status.heroMoving += toAdd;
-        if (core.status.heroMoving >= 8) {
-            clearInterval(core.interval.heroMoveInterval);
-            core.status.heroMoving = 0;
-            if (callback) callback();
-        }
-    }, core.values.moveSpeed / 8 * toAdd / core.status.replay.speed);
-}
+  }, ((core.values.moveSpeed / 8) * toAdd) / core.status.replay.speed);
+};
 
 ////// 每移动一格后执行的事件 //////
 control.prototype.moveOneStep = function (callback) {
-    return this.controldata.moveOneStep(callback);
-}
+  return this.controldata.moveOneStep(callback);
+};
 
 ////// 实际每一步的行走过程 //////
 control.prototype.moveAction = function (callback) {
-    if (core.status.heroMoving > 0) return;
-    var noPass = core.noPass(core.nextX(), core.nextY()), canMove = core.canMoveHero();
-    // 下一个点如果不能走
-    if (noPass || !canMove) return this._moveAction_noPass(canMove, callback);
-    this._moveAction_moving(callback);
-}
+  if (core.status.heroMoving > 0) return;
+  var noPass = core.noPass(core.nextX(), core.nextY()),
+    canMove = core.canMoveHero();
+  // 下一个点如果不能走
+  if (noPass || !canMove) return this._moveAction_noPass(canMove, callback);
+  this._moveAction_moving(callback);
+};
 
 control.prototype._moveAction_noPass = function (canMove, callback) {
-    core.status.route.push(core.getHeroLoc('direction'));
-    core.status.automaticRoute.moveStepBeforeStop = [];
-    core.status.automaticRoute.lastDirection = core.getHeroLoc('direction');
-    if (canMove) core.trigger(core.nextX(), core.nextY());
-    core.drawHero();
+  core.status.route.push(core.getHeroLoc("direction"));
+  core.status.automaticRoute.moveStepBeforeStop = [];
+  core.status.automaticRoute.lastDirection = core.getHeroLoc("direction");
+  if (canMove) core.trigger(core.nextX(), core.nextY());
+  core.drawHero();
 
-    if (core.status.automaticRoute.moveStepBeforeStop.length == 0) {
-        core.clearContinueAutomaticRoute();
-        core.stopAutomaticRoute();
-    }
-    if (callback) callback();
-}
+  if (core.status.automaticRoute.moveStepBeforeStop.length == 0) {
+    core.clearContinueAutomaticRoute();
+    core.stopAutomaticRoute();
+  }
+  if (callback) callback();
+};
 
 control.prototype._moveAction_moving = function (callback) {
-    core.setHeroMoveInterval(function () {
-        core.setHeroLoc('x', core.nextX(), true);
-        core.setHeroLoc('y', core.nextY(), true);
+  core.setHeroMoveInterval(function () {
+    core.setHeroLoc("x", core.nextX(), true);
+    core.setHeroLoc("y", core.nextY(), true);
 
-        var direction = core.getHeroLoc('direction');
-        core.control._moveAction_popAutomaticRoute();
-        core.status.route.push(direction);
+    var direction = core.getHeroLoc("direction");
+    core.control._moveAction_popAutomaticRoute();
+    core.status.route.push(direction);
 
-        core.moveOneStep();
-        core.checkRouteFolding();
-        if (callback) callback();
-    });
-}
+    core.moveOneStep();
+    core.checkRouteFolding();
+    if (callback) callback();
+  });
+};
 
 control.prototype._moveAction_popAutomaticRoute = function () {
-    var automaticRoute = core.status.automaticRoute;
-    // 检查自动寻路是否被弹出
-    if (automaticRoute.autoHeroMove) {
-        automaticRoute.movedStep++;
-        automaticRoute.lastDirection = core.getHeroLoc('direction');
-        if (automaticRoute.destStep == automaticRoute.movedStep) {
-            if (automaticRoute.autoStep == automaticRoute.autoStepRoutes.length) {
-                core.clearContinueAutomaticRoute();
-                core.stopAutomaticRoute();
-            }
-            else {
-                automaticRoute.movedStep = 0;
-                automaticRoute.destStep = automaticRoute.autoStepRoutes[automaticRoute.autoStep].step;
-                core.setHeroLoc('direction', automaticRoute.autoStepRoutes[automaticRoute.autoStep].direction);
-                core.status.automaticRoute.autoStep++;
-            }
-        }
+  var automaticRoute = core.status.automaticRoute;
+  // 检查自动寻路是否被弹出
+  if (automaticRoute.autoHeroMove) {
+    automaticRoute.movedStep++;
+    automaticRoute.lastDirection = core.getHeroLoc("direction");
+    if (automaticRoute.destStep == automaticRoute.movedStep) {
+      if (automaticRoute.autoStep == automaticRoute.autoStepRoutes.length) {
+        core.clearContinueAutomaticRoute();
+        core.stopAutomaticRoute();
+      } else {
+        automaticRoute.movedStep = 0;
+        automaticRoute.destStep =
+          automaticRoute.autoStepRoutes[automaticRoute.autoStep].step;
+        core.setHeroLoc(
+          "direction",
+          automaticRoute.autoStepRoutes[automaticRoute.autoStep].direction
+        );
+        core.status.automaticRoute.autoStep++;
+      }
     }
-}
+  }
+};
 
 ////// 让勇士开始移动 //////
 control.prototype.moveHero = function (direction, callback) {
-    // 如果正在移动,直接return
-    if (core.status.heroMoving != 0) return;
-    if (core.isset(direction))
-        core.setHeroLoc('direction', direction);
+  // 如果正在移动,直接return
+  if (core.status.heroMoving != 0) return;
+  if (core.isset(direction)) core.setHeroLoc("direction", direction);
 
-    if (callback) return this.moveAction(callback);
-    this._moveHero_moving();
-}
+  if (callback) return this.moveAction(callback);
+  this._moveHero_moving();
+};
 
 control.prototype._moveHero_moving = function () {
-    // ------ 我已经看不懂这个函数了,反正好用就行23333333
-    core.status.heroStop = false;
-    core.status.automaticRoute.moveDirectly = false;
-    var move = function () {
-        if (!core.status.heroStop) {
-            if (core.hasFlag('debug') && core.status.ctrlDown) {
-                if (core.status.heroMoving != 0) return;
-                // 检测是否穿出去
-                var nx = core.nextX(), ny = core.nextY();
-                if (nx < 0 || nx >= core.bigmap.width || ny < 0 || ny >= core.bigmap.height) return;
-                core.eventMoveHero([core.getHeroLoc('direction')], core.values.moveSpeed, move);
-            }
-            else {
-                core.moveAction();
-                setTimeout(move, 50);
-            }
-        }
+  // ------ 我已经看不懂这个函数了,反正好用就行23333333
+  core.status.heroStop = false;
+  core.status.automaticRoute.moveDirectly = false;
+  var move = function () {
+    if (!core.status.heroStop) {
+      if (core.hasFlag("debug") && core.status.ctrlDown) {
+        if (core.status.heroMoving != 0) return;
+        // 检测是否穿出去
+        var nx = core.nextX(),
+          ny = core.nextY();
+        if (
+          nx < 0 ||
+          nx >= core.bigmap.width ||
+          ny < 0 ||
+          ny >= core.bigmap.height
+        )
+          return;
+        core.eventMoveHero(
+          [core.getHeroLoc("direction")],
+          core.values.moveSpeed,
+          move
+        );
+      } else {
+        core.moveAction();
+        setTimeout(move, 50);
+      }
     }
-    move();
-}
+  };
+  move();
+};
 
 ////// 当前是否正在移动 //////
 control.prototype.isMoving = function () {
-    return !core.status.heroStop || core.status.heroMoving > 0;
-}
+  return !core.status.heroStop || core.status.heroMoving > 0;
+};
 
 ////// 停止勇士的一切行动,等待勇士行动结束后,再执行callback //////
 control.prototype.waitHeroToStop = function (callback) {
-    var lastDirection = core.status.automaticRoute.lastDirection;
-    core.stopAutomaticRoute();
-    core.clearContinueAutomaticRoute();
-    if (callback) {
-        core.status.replay.animate = true;
-        core.lockControl();
-        core.status.automaticRoute.moveDirectly = false;
-        setTimeout(function () {
-            core.status.replay.animate = false;
-            if (core.isset(lastDirection))
-                core.setHeroLoc('direction', lastDirection);
-            core.drawHero();
-            callback();
-        }, core.status.replay.speed == 24 ? 1 : 30);
-    }
-}
+  var lastDirection = core.status.automaticRoute.lastDirection;
+  core.stopAutomaticRoute();
+  core.clearContinueAutomaticRoute();
+  if (callback) {
+    core.status.replay.animate = true;
+    core.lockControl();
+    core.status.automaticRoute.moveDirectly = false;
+    setTimeout(
+      function () {
+        core.status.replay.animate = false;
+        if (core.isset(lastDirection))
+          core.setHeroLoc("direction", lastDirection);
+        core.drawHero();
+        callback();
+      },
+      core.status.replay.speed == 24 ? 1 : 30
+    );
+  }
+};
 
 ////// 转向 //////
 control.prototype.turnHero = function (direction) {
-    if (direction) {
-        core.setHeroLoc('direction', direction);
-        core.drawHero();
-        core.status.route.push("turn:" + direction);
-        return;
-    }
-    core.setHeroLoc('direction', core.turnDirection(':right'));
+  if (direction) {
+    core.setHeroLoc("direction", direction);
     core.drawHero();
-    core.status.route.push("turn");
-    core.checkRouteFolding();
-}
+    core.status.route.push("turn:" + direction);
+    return;
+  }
+  core.setHeroLoc("direction", core.turnDirection(":right"));
+  core.drawHero();
+  core.status.route.push("turn");
+  core.checkRouteFolding();
+};
 
 ////// 瞬间移动 //////
 control.prototype.moveDirectly = function (destX, destY, ignoreSteps) {
-    return this.controldata.moveDirectly(destX, destY, ignoreSteps);
-}
+  return this.controldata.moveDirectly(destX, destY, ignoreSteps);
+};
 
 ////// 尝试瞬间移动 //////
 control.prototype.tryMoveDirectly = function (destX, destY) {
-    if (this.nearHero(destX, destY)) return false;
-    var canMoveArray = core.maps.generateMovableArray();
-    var dirs = [[destX, destY], [destX - 1, destY, "right"], [destX, destY - 1, "down"], [destX, destY + 1, "up"], [destX + 1, destY, "left"]];
-    var canMoveDirectlyArray = core.canMoveDirectlyArray(dirs, canMoveArray);
+  if (this.nearHero(destX, destY)) return false;
+  var canMoveArray = core.maps.generateMovableArray();
+  var dirs = [
+    [destX, destY],
+    [destX - 1, destY, "right"],
+    [destX, destY - 1, "down"],
+    [destX, destY + 1, "up"],
+    [destX + 1, destY, "left"],
+  ];
+  var canMoveDirectlyArray = core.canMoveDirectlyArray(dirs, canMoveArray);
 
-    for (var i = 0; i < dirs.length; ++i) {
-        var d = dirs[i], dx = d[0], dy = d[1], dir = d[2];
-        if (dx < 0 || dx >= core.bigmap.width || dy < 0 || dy >= core.bigmap.height) continue;
-        if (dir && !core.inArray(canMoveArray[dx][dy], dir)) continue;
-        if (canMoveDirectlyArray[i] < 0) continue;
-        if (core.control.moveDirectly(dx, dy, canMoveDirectlyArray[i])) {
-            if (dir) core.moveHero(dir, function () { });
-            return true;
-        }
+  for (var i = 0; i < dirs.length; ++i) {
+    var d = dirs[i],
+      dx = d[0],
+      dy = d[1],
+      dir = d[2];
+    if (dx < 0 || dx >= core.bigmap.width || dy < 0 || dy >= core.bigmap.height)
+      continue;
+    if (dir && !core.inArray(canMoveArray[dx][dy], dir)) continue;
+    if (canMoveDirectlyArray[i] < 0) continue;
+    if (core.control.moveDirectly(dx, dy, canMoveDirectlyArray[i])) {
+      if (dir) core.moveHero(dir, function () {});
+      return true;
     }
-    return false;
-}
+  }
+  return false;
+};
 
 ////// 绘制勇士 //////
 control.prototype.drawHero = function (status, offset, frame) {
-    if (!core.isPlaying() || !core.status.floorId || core.status.gameOver) return;
-    var x = core.getHeroLoc('x'), y = core.getHeroLoc('y'), direction = core.getHeroLoc('direction');
-    status = status || 'stop';
-    if (!offset) offset = 0;
+  if (!core.isPlaying() || !core.status.floorId || core.status.gameOver) return;
+  var x = core.getHeroLoc("x"),
+    y = core.getHeroLoc("y"),
+    direction = core.getHeroLoc("direction");
+  status = status || "stop";
+  if (!offset) offset = 0;
 
-    var way = core.utils.scan2[direction];
-    var dx = way.x, dy = way.y;
-    var offsetX = typeof offset == 'number' ? dx * offset : (offset.x || 0);
-    var offsetY = typeof offset == 'number' ? dy * offset : (offset.y || 0);
-    offset = { x: offsetX, y: offsetY, offset: offset };
+  var way = core.utils.scan2[direction];
+  var dx = way.x,
+    dy = way.y;
+  var offsetX = typeof offset == "number" ? dx * offset : offset.x || 0;
+  var offsetY = typeof offset == "number" ? dy * offset : offset.y || 0;
+  offset = { x: offsetX, y: offsetY, offset: offset };
 
-    core.clearAutomaticRouteNode(x + dx, y + dy);
-    core.clearMap('hero');
-    core.status.heroCenter.px = 32 * x + offsetX + 16;
-    core.status.heroCenter.py = 32 * y + offsetY + 32 - core.material.icons.hero.height / 2;
+  core.clearAutomaticRouteNode(x + dx, y + dy);
+  core.clearMap("hero");
+  core.status.heroCenter.px = 32 * x + offsetX + 16;
+  core.status.heroCenter.py =
+    32 * y + offsetY + 32 - core.material.icons.hero.height / 2;
 
-    // 重置hero层画布
-    core.setGameCanvasTranslate('hero', 0, 0);
-    delete core.canvas.hero._px;
-    delete core.canvas.hero._py;
-    core.status.preview.enabled = false;
-    if (!core.hasFlag('__lockViewport__')) {
-        this._drawHero_updateViewport(x, y, offset);
-    }
+  // 重置hero层画布
+  core.setGameCanvasTranslate("hero", 0, 0);
+  delete core.canvas.hero._px;
+  delete core.canvas.hero._py;
+  core.status.preview.enabled = false;
+  if (!core.hasFlag("__lockViewport__")) {
+    this._drawHero_updateViewport(x, y, offset);
+  }
 
-    this._drawHero_draw(direction, x, y, status, offset, frame);
-}
+  this._drawHero_draw(direction, x, y, status, offset, frame);
+};
 
 control.prototype._drawHero_updateViewport = function (x, y, offset) {
-    core.bigmap.offsetX = core.clamp((x - core._HALF_WIDTH_) * 32 + offset.x, 0, Math.max(32 * core.bigmap.width - core._PX_, 0));
-    core.bigmap.offsetY = core.clamp((y - core._HALF_HEIGHT_) * 32 + offset.y, 0, Math.max(32 * core.bigmap.height - core._PY_, 0));
-    core.control.updateViewport();
-}
+  core.bigmap.offsetX = core.clamp(
+    (x - core._HALF_WIDTH_) * 32 + offset.x,
+    0,
+    Math.max(32 * core.bigmap.width - core._PX_, 0)
+  );
+  core.bigmap.offsetY = core.clamp(
+    (y - core._HALF_HEIGHT_) * 32 + offset.y,
+    0,
+    Math.max(32 * core.bigmap.height - core._PY_, 0)
+  );
+  core.control.updateViewport();
+};
 
-control.prototype._drawHero_draw = function (direction, x, y, status, offset, frame) {
-    offset = offset || { x: 0, y: 0, offset: 0, px: 0, py: 0 };
-    var opacity = core.setAlpha('hero', core.getFlag('__heroOpacity__', 1))
-    this._drawHero_getDrawObjs(direction, x, y, status, offset).forEach(function (block) {
-        core.drawImage('hero', block.img, (block.heroIcon[block.status] + (frame || 0)) % 4 * block.width,
-            block.heroIcon.loc * block.height, block.width, block.height,
-            block.posx + (32 - block.width) / 2, block.posy + 32 - block.height, block.width, block.height);
-    });
-    core.setAlpha('hero', opacity);
-}
+control.prototype._drawHero_draw = function (
+  direction,
+  x,
+  y,
+  status,
+  offset,
+  frame
+) {
+  offset = offset || { x: 0, y: 0, offset: 0, px: 0, py: 0 };
+  var opacity = core.setAlpha("hero", core.getFlag("__heroOpacity__", 1));
+  this._drawHero_getDrawObjs(direction, x, y, status, offset).forEach(function (
+    block
+  ) {
+    core.drawImage(
+      "hero",
+      block.img,
+      ((block.heroIcon[block.status] + (frame || 0)) % 4) * block.width,
+      block.heroIcon.loc * block.height,
+      block.width,
+      block.height,
+      block.posx + (32 - block.width) / 2,
+      block.posy + 32 - block.height,
+      block.width,
+      block.height
+    );
+  });
+  core.setAlpha("hero", opacity);
+};
 
-control.prototype._drawHero_getDrawObjs = function (direction, x, y, status, offset) {
-    var heroIconArr = core.material.icons.hero, drawObjs = [], index = 0;
-    drawObjs.push({
-        "img": core.material.images.hero,
-        "width": core.material.icons.hero.width || 32,
-        "height": core.material.icons.hero.height,
-        "heroIcon": heroIconArr[direction],
-        "posx": x * 32 - core.bigmap.offsetX + offset.x,
-        "posy": y * 32 - core.bigmap.offsetY + offset.y,
-        "status": status,
-        "index": index++,
+control.prototype._drawHero_getDrawObjs = function (
+  direction,
+  x,
+  y,
+  status,
+  offset
+) {
+  var heroIconArr = core.material.icons.hero,
+    drawObjs = [],
+    index = 0;
+  drawObjs.push({
+    img: core.material.images.hero,
+    width: core.material.icons.hero.width || 32,
+    height: core.material.icons.hero.height,
+    heroIcon: heroIconArr[direction],
+    posx: x * 32 - core.bigmap.offsetX + offset.x,
+    posy: y * 32 - core.bigmap.offsetY + offset.y,
+    status: status,
+    index: index++,
+  });
+  if (typeof offset.offset == "number") {
+    core.status.hero.followers.forEach(function (t) {
+      drawObjs.push({
+        img: core.material.images.images[t.name],
+        width: core.material.images.images[t.name].width / 4,
+        height: core.material.images.images[t.name].height / 4,
+        heroIcon: heroIconArr[t.direction],
+        posx:
+          32 * t.x -
+          core.bigmap.offsetX +
+          (t.stop
+            ? 0
+            : core.utils.scan2[t.direction].x * Math.abs(offset.offset)),
+        posy:
+          32 * t.y -
+          core.bigmap.offsetY +
+          (t.stop
+            ? 0
+            : core.utils.scan2[t.direction].y * Math.abs(offset.offset)),
+        status: t.stop ? "stop" : status,
+        index: index++,
+      });
     });
-    if (typeof offset.offset == 'number') {
-        core.status.hero.followers.forEach(function (t) {
-            drawObjs.push({
-                "img": core.material.images.images[t.name],
-                "width": core.material.images.images[t.name].width / 4,
-                "height": core.material.images.images[t.name].height / 4,
-                "heroIcon": heroIconArr[t.direction],
-                "posx": 32 * t.x - core.bigmap.offsetX + (t.stop ? 0 : core.utils.scan2[t.direction].x * Math.abs(offset.offset)),
-                "posy": 32 * t.y - core.bigmap.offsetY + (t.stop ? 0 : core.utils.scan2[t.direction].y * Math.abs(offset.offset)),
-                "status": t.stop ? "stop" : status,
-                "index": index++
-            });
-        });
+  }
+  return drawObjs.sort(function (a, b) {
+    return a.posy == b.posy ? b.index - a.index : a.posy - b.posy;
+  });
+};
+
+control.prototype.setHeroOpacity = function (
+  opacity,
+  moveMode,
+  time,
+  callback
+) {
+  time = time || 0;
+  if (time == 0) {
+    core.setFlag("__heroOpacity__", opacity);
+    core.drawHero();
+    if (callback) callback();
+    return;
+  }
+  time /= Math.max(core.status.replay.speed, 1);
+
+  var fromOpacity = core.getFlag("__heroOpacity__", 1);
+  var step = 0,
+    steps = parseInt(time / 10);
+  if (steps <= 0) steps = 1;
+  var moveFunc = core.applyEasing(moveMode);
+
+  var animate = setInterval(function () {
+    step++;
+    core.setFlag(
+      "__heroOpacity__",
+      fromOpacity + (opacity - fromOpacity) * moveFunc(step / steps)
+    );
+    core.drawHero();
+    if (step == steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      if (callback) callback();
     }
-    return drawObjs.sort(function (a, b) {
-        return a.posy == b.posy ? b.index - a.index : a.posy - b.posy;
-    });
-}
+  }, 10);
 
-control.prototype.setHeroOpacity = function (opacity, moveMode, time, callback) {
-    time = time || 0;
-    if (time == 0) {
-        core.setFlag('__heroOpacity__', opacity);
-        core.drawHero();
-        if (callback) callback();
-        return;
-    }
-    time /= Math.max(core.status.replay.speed, 1)
-
-    var fromOpacity = core.getFlag('__heroOpacity__', 1);
-    var step = 0, steps = parseInt(time / 10);
-    if (steps <= 0) steps = 1;
-    var moveFunc = core.applyEasing(moveMode);
-
-    var animate = setInterval(function () {
-        step++;
-        core.setFlag('__heroOpacity__', fromOpacity + (opacity - fromOpacity) * moveFunc(step / steps));
-        core.drawHero();
-        if (step == steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            if (callback) callback();
-        }
-    }, 10);
-
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = callback;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = callback;
+};
 
 // ------ 画布、位置、阻激夹域,显伤 ------ //
 
 ////// 设置画布偏移
 control.prototype.setGameCanvasTranslate = function (canvas, x, y) {
-    var c = core.dom.gameCanvas[canvas];
-    x = x * core.domStyle.scale;
-    y = y * core.domStyle.scale;
-    c.style.transform = 'translate(' + x + 'px,' + y + 'px)';
-    c.style.webkitTransform = 'translate(' + x + 'px,' + y + 'px)';
-    c.style.OTransform = 'translate(' + x + 'px,' + y + 'px)';
-    c.style.MozTransform = 'translate(' + x + 'px,' + y + 'px)';
-    if (main.mode === 'editor' && editor.isMobile) {
-        c.style.transform = 'translate(' + (x / core._PX_ * 96) + 'vw,' + (y / core._PY_ * 96) + 'vw)';
-        c.style.webkitTransform = 'translate(' + (x / core._PX_ * 96) + 'vw,' + (y / core._PY_ * 96) + 'vw)';
-        c.style.OTransform = 'translate(' + (x / core._PX_ * 96) + 'vw,' + (y / core._PY_ * 96) + 'vw)';
-        c.style.MozTransform = 'translate(' + (x / core._PX_ * 96) + 'vw,' + (y / core._PY_ * 96) + 'vw)';
-    }
+  var c = core.dom.gameCanvas[canvas];
+  x = x * core.domStyle.scale;
+  y = y * core.domStyle.scale;
+  c.style.transform = "translate(" + x + "px," + y + "px)";
+  c.style.webkitTransform = "translate(" + x + "px," + y + "px)";
+  c.style.OTransform = "translate(" + x + "px," + y + "px)";
+  c.style.MozTransform = "translate(" + x + "px," + y + "px)";
+  if (main.mode === "editor" && editor.isMobile) {
+    c.style.transform =
+      "translate(" +
+      (x / core._PX_) * 96 +
+      "vw," +
+      (y / core._PY_) * 96 +
+      "vw)";
+    c.style.webkitTransform =
+      "translate(" +
+      (x / core._PX_) * 96 +
+      "vw," +
+      (y / core._PY_) * 96 +
+      "vw)";
+    c.style.OTransform =
+      "translate(" +
+      (x / core._PX_) * 96 +
+      "vw," +
+      (y / core._PY_) * 96 +
+      "vw)";
+    c.style.MozTransform =
+      "translate(" +
+      (x / core._PX_) * 96 +
+      "vw," +
+      (y / core._PY_) * 96 +
+      "vw)";
+  }
 };
 
 ////// 加减画布偏移
 control.prototype.addGameCanvasTranslate = function (x, y) {
-    for (var ii = 0, canvas; canvas = core.dom.gameCanvas[ii]; ii++) {
-        var id = canvas.getAttribute('id');
-        if (id == 'ui' || id == 'data') continue; // UI层和data层不移动
-        var offsetX = x, offsetY = y;
-        if (core.bigmap.canvas.indexOf(id) >= 0) {
-            if (core.bigmap.v2) {
-                offsetX -= (core.bigmap.offsetX - 32 * core.bigmap.posX) + 32;
-                offsetY -= (core.bigmap.offsetY - 32 * core.bigmap.posY) + 32;
-            } else {
-                offsetX -= core.bigmap.offsetX;
-                offsetY -= core.bigmap.offsetY;
-            }
-        }
-        core.control.setGameCanvasTranslate(id, offsetX, offsetY);
+  for (var ii = 0, canvas; (canvas = core.dom.gameCanvas[ii]); ii++) {
+    var id = canvas.getAttribute("id");
+    if (id == "ui" || id == "data") continue; // UI层和data层不移动
+    var offsetX = x,
+      offsetY = y;
+    if (core.bigmap.canvas.indexOf(id) >= 0) {
+      if (core.bigmap.v2) {
+        offsetX -= core.bigmap.offsetX - 32 * core.bigmap.posX + 32;
+        offsetY -= core.bigmap.offsetY - 32 * core.bigmap.posY + 32;
+      } else {
+        offsetX -= core.bigmap.offsetX;
+        offsetY -= core.bigmap.offsetY;
+      }
     }
-}
+    core.control.setGameCanvasTranslate(id, offsetX, offsetY);
+  }
+};
 
 ////// 更新视野范围 //////
 control.prototype.updateViewport = function () {
-    // 当前是否应该重绘?
-    if (core.bigmap.v2) {
-        if (core.bigmap.offsetX >= core.bigmap.posX * 32 + 32
-            || core.bigmap.offsetX <= core.bigmap.posX * 32 - 32
-            || core.bigmap.offsetY >= core.bigmap.posY * 32 + 32
-            || core.bigmap.offsetY <= core.bigmap.posY * 32 - 32) {
-            core.bigmap.posX = parseInt(core.bigmap.offsetX / 32);
-            core.bigmap.posY = parseInt(core.bigmap.offsetY / 32);
-            core.redrawMap();
-        }
-    } else {
-        core.bigmap.posX = core.bigmap.posY = 0;
+  // 当前是否应该重绘?
+  if (core.bigmap.v2) {
+    if (
+      core.bigmap.offsetX >= core.bigmap.posX * 32 + 32 ||
+      core.bigmap.offsetX <= core.bigmap.posX * 32 - 32 ||
+      core.bigmap.offsetY >= core.bigmap.posY * 32 + 32 ||
+      core.bigmap.offsetY <= core.bigmap.posY * 32 - 32
+    ) {
+      core.bigmap.posX = parseInt(core.bigmap.offsetX / 32);
+      core.bigmap.posY = parseInt(core.bigmap.offsetY / 32);
+      core.redrawMap();
     }
-    var offsetX = core.bigmap.v2 ? -(core.bigmap.offsetX - 32 * core.bigmap.posX) - 32 : -core.bigmap.offsetX;
-    var offsetY = core.bigmap.v2 ? -(core.bigmap.offsetY - 32 * core.bigmap.posY) - 32 : -core.bigmap.offsetY;
+  } else {
+    core.bigmap.posX = core.bigmap.posY = 0;
+  }
+  var offsetX = core.bigmap.v2
+    ? -(core.bigmap.offsetX - 32 * core.bigmap.posX) - 32
+    : -core.bigmap.offsetX;
+  var offsetY = core.bigmap.v2
+    ? -(core.bigmap.offsetY - 32 * core.bigmap.posY) - 32
+    : -core.bigmap.offsetY;
 
-    core.bigmap.canvas.forEach(function (cn) {
-        core.control.setGameCanvasTranslate(cn, offsetX, offsetY);
-    });
-    // ------ 路线
-    core.relocateCanvas('route', core.status.automaticRoute.offsetX - core.bigmap.offsetX, core.status.automaticRoute.offsetY - core.bigmap.offsetY);
-    // ------ 所有的大怪物也都需要重定位
-    for (var one in core.dymCanvas) {
-        if (one.startsWith('_bigImage_')) {
-            var ox = core.dymCanvas[one].canvas.getAttribute('_ox');
-            var oy = core.dymCanvas[one].canvas.getAttribute('_oy');
-            if (ox != null && oy != null) {
-                core.relocateCanvas(one, parseInt(ox) - core.bigmap.offsetX, parseInt(oy) - core.bigmap.offsetY);
-            }
-        }
+  core.bigmap.canvas.forEach(function (cn) {
+    core.control.setGameCanvasTranslate(cn, offsetX, offsetY);
+  });
+  // ------ 路线
+  core.relocateCanvas(
+    "route",
+    core.status.automaticRoute.offsetX - core.bigmap.offsetX,
+    core.status.automaticRoute.offsetY - core.bigmap.offsetY
+  );
+  // ------ 所有的大怪物也都需要重定位
+  for (var one in core.dymCanvas) {
+    if (one.startsWith("_bigImage_")) {
+      var ox = core.dymCanvas[one].canvas.getAttribute("_ox");
+      var oy = core.dymCanvas[one].canvas.getAttribute("_oy");
+      if (ox != null && oy != null) {
+        core.relocateCanvas(
+          one,
+          parseInt(ox) - core.bigmap.offsetX,
+          parseInt(oy) - core.bigmap.offsetY
+        );
+      }
     }
-
-}
+  }
+};
 
 ////// 设置视野范围 //////
 control.prototype.setViewport = function (px, py) {
-    var originOffsetX = core.bigmap.offsetX, originOffsetY = core.bigmap.offsetY;
-    core.bigmap.offsetX = core.clamp(px, 0, 32 * core.bigmap.width - core._PX_);
-    core.bigmap.offsetY = core.clamp(py, 0, 32 * core.bigmap.height - core._PY_);
-    this.updateViewport();
-    // ------ hero层也需要!
-    var px = parseFloat(core.canvas.hero._px) || 0;
-    var py = parseFloat(core.canvas.hero._py) || 0;
-    px += originOffsetX - core.bigmap.offsetX;
-    py += originOffsetY - core.bigmap.offsetY;
-    core.control.setGameCanvasTranslate('hero', px, py);
-    core.canvas.hero._px = px;
-    core.canvas.hero._py = py;
-}
+  var originOffsetX = core.bigmap.offsetX,
+    originOffsetY = core.bigmap.offsetY;
+  core.bigmap.offsetX = core.clamp(px, 0, 32 * core.bigmap.width - core._PX_);
+  core.bigmap.offsetY = core.clamp(py, 0, 32 * core.bigmap.height - core._PY_);
+  this.updateViewport();
+  // ------ hero层也需要!
+  var px = parseFloat(core.canvas.hero._px) || 0;
+  var py = parseFloat(core.canvas.hero._py) || 0;
+  px += originOffsetX - core.bigmap.offsetX;
+  py += originOffsetY - core.bigmap.offsetY;
+  core.control.setGameCanvasTranslate("hero", px, py);
+  core.canvas.hero._px = px;
+  core.canvas.hero._py = py;
+};
 
 ////// 移动视野范围 //////
 control.prototype.moveViewport = function (x, y, moveMode, time, callback) {
-    time = time || 0;
-    time /= Math.max(core.status.replay.speed, 1)
-    var per_time = 10, step = 0, steps = parseInt(time / per_time);
-    if (steps <= 0) {
-        this.setViewport(32 * x, 32 * y);
-        if (callback) callback();
-        return;
+  time = time || 0;
+  time /= Math.max(core.status.replay.speed, 1);
+  var per_time = 10,
+    step = 0,
+    steps = parseInt(time / per_time);
+  if (steps <= 0) {
+    this.setViewport(32 * x, 32 * y);
+    if (callback) callback();
+    return;
+  }
+  var px = core.clamp(32 * x, 0, 32 * core.bigmap.width - core._PX_);
+  var py = core.clamp(32 * y, 0, 32 * core.bigmap.height - core._PY_);
+  var cx = core.bigmap.offsetX;
+  var cy = core.bigmap.offsetY;
+  var moveFunc = core.applyEasing(moveMode);
+
+  var animate = window.setInterval(function () {
+    step++;
+    core.setViewport(
+      cx + moveFunc(step / steps) * (px - cx),
+      cy + moveFunc(step / steps) * (py - cy)
+    );
+    if (step == steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      core.setViewport(px, py);
+      if (callback) callback();
     }
-    var px = core.clamp(32 * x, 0, 32 * core.bigmap.width - core._PX_);
-    var py = core.clamp(32 * y, 0, 32 * core.bigmap.height - core._PY_);
-    var cx = core.bigmap.offsetX;
-    var cy = core.bigmap.offsetY;
-    var moveFunc = core.applyEasing(moveMode);
+  }, per_time);
 
-    var animate = window.setInterval(function () {
-        step++;
-        core.setViewport(cx + moveFunc(step / steps) * (px - cx), cy + moveFunc(step / steps) * (py - cy));
-        if (step == steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            core.setViewport(px, py);
-            if (callback) callback();
-        }
-    }, per_time);
-
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = callback;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = callback;
+};
 
 ////// 获得勇士面对位置的x坐标 //////
 control.prototype.nextX = function (n) {
-    if (n == null) n = 1;
-    return core.getHeroLoc('x') + core.utils.scan[core.getHeroLoc('direction')].x * n;
-}
+  if (n == null) n = 1;
+  return (
+    core.getHeroLoc("x") + core.utils.scan[core.getHeroLoc("direction")].x * n
+  );
+};
 
 ////// 获得勇士面对位置的y坐标 //////
 control.prototype.nextY = function (n) {
-    if (n == null) n = 1;
-    return core.getHeroLoc('y') + core.utils.scan[core.getHeroLoc('direction')].y * n;
-}
+  if (n == null) n = 1;
+  return (
+    core.getHeroLoc("y") + core.utils.scan[core.getHeroLoc("direction")].y * n
+  );
+};
 
 ////// 某个点是否在勇士旁边 //////
 control.prototype.nearHero = function (x, y, n) {
-    if (n == null) n = 1;
-    return Math.abs(x - core.getHeroLoc('x')) + Math.abs(y - core.getHeroLoc('y')) <= n;
-}
+  if (n == null) n = 1;
+  return (
+    Math.abs(x - core.getHeroLoc("x")) + Math.abs(y - core.getHeroLoc("y")) <= n
+  );
+};
 
 ////// 聚集跟随者 //////
 control.prototype.gatherFollowers = function () {
-    var x = core.getHeroLoc('x'), y = core.getHeroLoc('y'), dir = core.getHeroLoc('direction');
-    core.status.hero.followers.forEach(function (t) {
-        t.x = x;
-        t.y = y;
-        t.stop = true;
-        t.direction = dir;
-    });
-}
+  var x = core.getHeroLoc("x"),
+    y = core.getHeroLoc("y"),
+    dir = core.getHeroLoc("direction");
+  core.status.hero.followers.forEach(function (t) {
+    t.x = x;
+    t.y = y;
+    t.stop = true;
+    t.direction = dir;
+  });
+};
 
 ////// 更新跟随者坐标 //////
 control.prototype.updateFollowers = function () {
-    core.status.hero.followers.forEach(function (t) {
-        if (!t.stop) {
-            t.x += core.utils.scan2[t.direction].x;
-            t.y += core.utils.scan2[t.direction].y;
-        }
-    })
+  core.status.hero.followers.forEach(function (t) {
+    if (!t.stop) {
+      t.x += core.utils.scan2[t.direction].x;
+      t.y += core.utils.scan2[t.direction].y;
+    }
+  });
 
-    var nowx = core.getHeroLoc('x'), nowy = core.getHeroLoc('y');
-    core.status.hero.followers.forEach(function (t) {
-        t.stop = true;
-        var dx = nowx - t.x, dy = nowy - t.y;
-        for (var dir in core.utils.scan2) {
-            if (core.utils.scan2[dir].x == dx && core.utils.scan2[dir].y == dy) {
-                t.stop = false;
-                t.direction = dir;
-            }
-        }
-        nowx = t.x; nowy = t.y;
-    })
-}
+  var nowx = core.getHeroLoc("x"),
+    nowy = core.getHeroLoc("y");
+  core.status.hero.followers.forEach(function (t) {
+    t.stop = true;
+    var dx = nowx - t.x,
+      dy = nowy - t.y;
+    for (var dir in core.utils.scan2) {
+      if (core.utils.scan2[dir].x == dx && core.utils.scan2[dir].y == dy) {
+        t.stop = false;
+        t.direction = dir;
+      }
+    }
+    nowx = t.x;
+    nowy = t.y;
+  });
+};
 
 ////// 瞬移更新跟随者坐标 //////
 control.prototype._moveDirectyFollowers = function (x, y) {
-    var route = core.automaticRoute(x, y);
-    if (route.length == 0) route = [{ x: x, y: y, direction: core.getHeroLoc('direction') }];
+  var route = core.automaticRoute(x, y);
+  if (route.length == 0)
+    route = [{ x: x, y: y, direction: core.getHeroLoc("direction") }];
 
-    var nowx = x, nowy = y;
-    for (var i = 0; i < core.status.hero.followers.length; ++i) {
-        var t = core.status.hero.followers[i];
-        var index = route.length - i - 2;
-        if (index < 0) index = 0;
-        t.stop = true;
-        t.x = route[index].x;
-        t.y = route[index].y;
-        t.direction = route[index].direction;
-        var dx = nowx - t.x, dy = nowy - t.y;
-        for (var dir in core.utils.scan2) {
-            if (core.utils.scan2[dir].x == dx && core.utils.scan2[dir].y == dy) {
-                t.stop = false;
-                t.direction = dir;
-            }
-        }
-        nowx = t.x; nowy = t.y;
+  var nowx = x,
+    nowy = y;
+  for (var i = 0; i < core.status.hero.followers.length; ++i) {
+    var t = core.status.hero.followers[i];
+    var index = route.length - i - 2;
+    if (index < 0) index = 0;
+    t.stop = true;
+    t.x = route[index].x;
+    t.y = route[index].y;
+    t.direction = route[index].direction;
+    var dx = nowx - t.x,
+      dy = nowy - t.y;
+    for (var dir in core.utils.scan2) {
+      if (core.utils.scan2[dir].x == dx && core.utils.scan2[dir].y == dy) {
+        t.stop = false;
+        t.direction = dir;
+      }
     }
-}
+    nowx = t.x;
+    nowy = t.y;
+  }
+};
 
 ////// 更新领域、夹击、阻击的伤害地图 //////
 control.prototype.updateCheckBlock = function (floorId) {
-    return this.controldata.updateCheckBlock(floorId);
-}
+  return this.controldata.updateCheckBlock(floorId);
+};
 
 ////// 检查并执行领域、夹击、阻击事件 //////
 control.prototype.checkBlock = function () {
-    var x = core.getHeroLoc('x'), y = core.getHeroLoc('y'), loc = x + "," + y;
-    var damage = core.status.checkBlock.damage[loc];
-    if (damage) {
-        core.status.hero.hp -= damage;
-        var text = (Object.keys(core.status.checkBlock.type[loc] || {}).join(",")) || "伤害";
-        core.drawTip("受到" + text + damage + "点");
-        core.drawHeroAnimate("zone");
-        this._checkBlock_disableQuickShop();
-        core.status.hero.statistics.extraDamage += damage;
-        if (core.status.hero.hp <= 0) {
-            core.status.hero.hp = 0;
-            core.updateStatusBar(false, true);
-            core.events.lose();
-            return;
-        } else {
-            core.updateStatusBar(false, true);
-        }
+  var x = core.getHeroLoc("x"),
+    y = core.getHeroLoc("y"),
+    loc = x + "," + y;
+  var damage = core.status.checkBlock.damage[loc];
+  if (damage) {
+    core.status.hero.hp -= damage;
+    var text =
+      Object.keys(core.status.checkBlock.type[loc] || {}).join(",") || "伤害";
+    core.drawTip("受到" + text + damage + "点");
+    core.drawHeroAnimate("zone");
+    this._checkBlock_disableQuickShop();
+    core.status.hero.statistics.extraDamage += damage;
+    if (core.status.hero.hp <= 0) {
+      core.status.hero.hp = 0;
+      core.updateStatusBar(false, true);
+      core.events.lose();
+      return;
+    } else {
+      core.updateStatusBar(false, true);
     }
-    this._checkBlock_ambush(core.status.checkBlock.ambush[loc]);
-    this._checkBlock_repulse(core.status.checkBlock.repulse[loc]);
-}
+  }
+  this._checkBlock_ambush(core.status.checkBlock.ambush[loc]);
+  this._checkBlock_repulse(core.status.checkBlock.repulse[loc]);
+};
 
 control.prototype._checkBlock_disableQuickShop = function () {
-    // 禁用快捷商店
-    if (core.flags.disableShopOnDamage) {
-        Object.keys(core.status.shops).forEach(function (shopId) {
-            core.setShopVisited(shopId, false);
-        });
-    }
-}
+  // 禁用快捷商店
+  if (core.flags.disableShopOnDamage) {
+    Object.keys(core.status.shops).forEach(function (shopId) {
+      core.setShopVisited(shopId, false);
+    });
+  }
+};
 
 ////// 阻击 //////
 control.prototype._checkBlock_repulse = function (repulse) {
-    if (!repulse || repulse.length == 0) return;
-    var actions = [];
-    repulse.forEach(function (t) {
-        actions.push({ "type": "move", "loc": [t[0], t[1]], "steps": [t[3]], "time": 250, "keep": true, "async": true });
+  if (!repulse || repulse.length == 0) return;
+  var actions = [];
+  repulse.forEach(function (t) {
+    actions.push({
+      type: "move",
+      loc: [t[0], t[1]],
+      steps: [t[3]],
+      time: 250,
+      keep: true,
+      async: true,
     });
-    actions.push({ "type": "waitAsync" });
-    core.insertAction(actions);
-}
+  });
+  actions.push({ type: "waitAsync" });
+  core.insertAction(actions);
+};
 
 ////// 捕捉 //////
 control.prototype._checkBlock_ambush = function (ambush) {
-    if (!ambush || ambush.length == 0) return;
-    // 捕捉效果
-    var actions = [];
-    ambush.forEach(function (t) {
-        actions.push({ "type": "move", "loc": [t[0], t[1]], "steps": [t[3]], "time": 250, "keep": false, "async": true });
+  if (!ambush || ambush.length == 0) return;
+  // 捕捉效果
+  var actions = [];
+  ambush.forEach(function (t) {
+    actions.push({
+      type: "move",
+      loc: [t[0], t[1]],
+      steps: [t[3]],
+      time: 250,
+      keep: false,
+      async: true,
     });
-    actions.push({ "type": "waitAsync" });
-    // 强制战斗
-    ambush.forEach(function (t) {
-        actions.push({
-            "type": "function", "function": "function() { " +
-                "core.battle('" + t[2] + "', " + t[0] + "," + t[1] + ", true, core.doAction); " +
-                "}", "async": true
-        });
+  });
+  actions.push({ type: "waitAsync" });
+  // 强制战斗
+  ambush.forEach(function (t) {
+    actions.push({
+      type: "function",
+      function:
+        "function() { " +
+        "core.battle('" +
+        t[2] +
+        "', " +
+        t[0] +
+        "," +
+        t[1] +
+        ", true, core.doAction); " +
+        "}",
+      async: true,
     });
-    core.insertAction(actions);
-}
+  });
+  core.insertAction(actions);
+};
 
 ////// 更新全地图显伤 //////
 control.prototype.updateDamage = function (floorId, ctx) {
-    floorId = floorId || core.status.floorId;
-    if (!floorId || core.status.gameOver || main.mode != 'play') return;
-    var onMap = ctx == null;
+  floorId = floorId || core.status.floorId;
+  if (!floorId || core.status.gameOver || main.mode != "play") return;
+  var onMap = ctx == null;
 
-    // 没有怪物手册
-    if (!core.hasItem('book')) return;
-    core.status.damage.posX = core.bigmap.posX;
-    core.status.damage.posY = core.bigmap.posY;
-    if (!onMap) {
-        var width = core.floors[floorId].width, height = core.floors[floorId].height;
-        // 地图过大的缩略图不绘制显伤
-        if (width * height > core.bigmap.threshold) return;
-    }
-    this._updateDamage_damage(floorId, onMap);
-    this._updateDamage_extraDamage(floorId, onMap);
-    this.drawDamage(ctx);
-}
+  // 没有怪物手册
+  if (!core.hasItem("book")) return;
+  core.status.damage.posX = core.bigmap.posX;
+  core.status.damage.posY = core.bigmap.posY;
+  if (!onMap) {
+    var width = core.floors[floorId].width,
+      height = core.floors[floorId].height;
+    // 地图过大的缩略图不绘制显伤
+    if (width * height > core.bigmap.threshold) return;
+  }
+  this._updateDamage_damage(floorId, onMap);
+  this._updateDamage_extraDamage(floorId, onMap);
+  this.drawDamage(ctx);
+};
 
 control.prototype._updateDamage_damage = function (floorId, onMap) {
-    core.status.damage.data = [];
-    if (!core.flags.displayEnemyDamage && !core.flags.displayExtraDamage) return;
+  core.status.damage.data = [];
+  if (!core.flags.displayEnemyDamage && !core.flags.displayExtraDamage) return;
 
-    core.extractBlocks(floorId);
-    core.status.maps[floorId].blocks.forEach(function (block) {
-        var x = block.x, y = block.y;
+  core.extractBlocks(floorId);
+  core.status.maps[floorId].blocks.forEach(function (block) {
+    var x = block.x,
+      y = block.y;
 
-        // v2优化,只绘制范围内的部分
-        if (onMap && core.bigmap.v2) {
-            if (x < core.bigmap.posX - core.bigmap.extend || x > core.bigmap.posX + core._WIDTH_ + core.bigmap.extend
-                || y < core.bigmap.posY - core.bigmap.extend || y > core.bigmap.posY + core._HEIGHT_ + core.bigmap.extend) {
-                return;
-            }
-        }
+    // v2优化,只绘制范围内的部分
+    if (onMap && core.bigmap.v2) {
+      if (
+        x < core.bigmap.posX - core.bigmap.extend ||
+        x > core.bigmap.posX + core._WIDTH_ + core.bigmap.extend ||
+        y < core.bigmap.posY - core.bigmap.extend ||
+        y > core.bigmap.posY + core._HEIGHT_ + core.bigmap.extend
+      ) {
+        return;
+      }
+    }
 
-        if (!block.disable && block.event.cls.indexOf('enemy') == 0 && block.event.displayDamage !== false) {
-            if (core.flags.displayEnemyDamage) {
-                var damageString = core.enemys.getDamageString(block.event.id, x, y, floorId);
-                core.status.damage.data.push({ text: damageString.damage, px: 32 * x + 1, py: 32 * (y + 1) - 1, color: damageString.color });
-            }
-            if (core.flags.displayCritical) {
-                var critical = core.enemys.nextCriticals(block.event.id, 1, x, y, floorId);
-                critical = core.formatBigNumber((critical[0] || [])[0], true);
-                if (critical == '???') critical = '?';
-                core.status.damage.data.push({ text: critical, px: 32 * x + 1, py: 32 * (y + 1) - 11, color: '#FFFFFF' });
-            }
-        }
-    });
-}
+    if (
+      !block.disable &&
+      block.event.cls.indexOf("enemy") == 0 &&
+      block.event.displayDamage !== false
+    ) {
+      if (core.flags.displayEnemyDamage) {
+        var damageString = core.enemys.getDamageString(
+          block.event.id,
+          x,
+          y,
+          floorId
+        );
+        core.status.damage.data.push({
+          text: damageString.damage,
+          px: 32 * x + 1,
+          py: 32 * (y + 1) - 1,
+          color: damageString.color,
+        });
+      }
+      if (core.flags.displayCritical) {
+        var critical = core.enemys.nextCriticals(
+          block.event.id,
+          1,
+          x,
+          y,
+          floorId
+        );
+        critical = core.formatBigNumber((critical[0] || [])[0], true);
+        if (critical == "???") critical = "?";
+        core.status.damage.data.push({
+          text: critical,
+          px: 32 * x + 1,
+          py: 32 * (y + 1) - 11,
+          color: "#FFFFFF",
+        });
+      }
+    }
+  });
+};
 
 control.prototype._updateDamage_extraDamage = function (floorId, onMap) {
-    core.status.damage.extraData = [];
-    if (!core.flags.displayExtraDamage) return;
+  core.status.damage.extraData = [];
+  if (!core.flags.displayExtraDamage) return;
 
-    var width = core.floors[floorId].width, height = core.floors[floorId].height;
-    var startX = onMap && core.bigmap.v2 ? Math.max(0, core.bigmap.posX - core.bigmap.extend) : 0;
-    var endX = onMap && core.bigmap.v2 ? Math.min(width, core.bigmap.posX + core._WIDTH_ + core.bigmap.extend + 1) : width;
-    var startY = onMap && core.bigmap.v2 ? Math.max(0, core.bigmap.posY - core.bigmap.extend) : 0;
-    var endY = onMap && core.bigmap.v2 ? Math.min(height, core.bigmap.posY + core._HEIGHT_ + core.bigmap.extend + 1) : height;
+  var width = core.floors[floorId].width,
+    height = core.floors[floorId].height;
+  var startX =
+    onMap && core.bigmap.v2
+      ? Math.max(0, core.bigmap.posX - core.bigmap.extend)
+      : 0;
+  var endX =
+    onMap && core.bigmap.v2
+      ? Math.min(
+          width,
+          core.bigmap.posX + core._WIDTH_ + core.bigmap.extend + 1
+        )
+      : width;
+  var startY =
+    onMap && core.bigmap.v2
+      ? Math.max(0, core.bigmap.posY - core.bigmap.extend)
+      : 0;
+  var endY =
+    onMap && core.bigmap.v2
+      ? Math.min(
+          height,
+          core.bigmap.posY + core._HEIGHT_ + core.bigmap.extend + 1
+        )
+      : height;
 
-    for (var x = startX; x < endX; x++) {
-        for (var y = startY; y < endY; y++) {
-            var alpha = 1;
-            if (core.noPass(x, y, floorId)) {
-                if (core.flags.extraDamageType == 2) alpha = 0;
-                else if (core.flags.extraDamageType == 1) alpha = 0.6;
-            }
-            var damage = core.status.checkBlock.damage[x + "," + y] || 0;
-            if (damage > 0) { // 该点伤害
-                damage = core.formatBigNumber(damage, true);
-                core.status.damage.extraData.push({ text: damage, px: 32 * x + 16, py: 32 * (y + 1) - 14, color: '#ffaa33', alpha: alpha });
-            }
-            else { // 检查捕捉
-                if (core.status.checkBlock.ambush[x + "," + y]) {
-                    core.status.damage.extraData.push({ text: '!', px: 32 * x + 16, py: 32 * (y + 1) - 14, color: '#ffaa33', alpha: alpha });
-                }
-            }
+  for (var x = startX; x < endX; x++) {
+    for (var y = startY; y < endY; y++) {
+      var alpha = 1;
+      if (core.noPass(x, y, floorId)) {
+        if (core.flags.extraDamageType == 2) alpha = 0;
+        else if (core.flags.extraDamageType == 1) alpha = 0.6;
+      }
+      var damage = core.status.checkBlock.damage[x + "," + y] || 0;
+      if (damage > 0) {
+        // 该点伤害
+        damage = core.formatBigNumber(damage, true);
+        core.status.damage.extraData.push({
+          text: damage,
+          px: 32 * x + 16,
+          py: 32 * (y + 1) - 14,
+          color: "#ffaa33",
+          alpha: alpha,
+        });
+      } else {
+        // 检查捕捉
+        if (core.status.checkBlock.ambush[x + "," + y]) {
+          core.status.damage.extraData.push({
+            text: "!",
+            px: 32 * x + 16,
+            py: 32 * (y + 1) - 14,
+            color: "#ffaa33",
+            alpha: alpha,
+          });
         }
+      }
     }
-}
+  }
+};
 
 ////// 重绘地图显伤 //////
 control.prototype.drawDamage = function (ctx) {
-    if (core.status.gameOver || !core.status.damage || main.mode != 'play') return;
-    var onMap = false;
-    if (ctx == null) {
-        ctx = core.canvas.damage;
-        core.clearMap('damage');
-        onMap = true;
-    }
+  if (core.status.gameOver || !core.status.damage || main.mode != "play")
+    return;
+  var onMap = false;
+  if (ctx == null) {
+    ctx = core.canvas.damage;
+    core.clearMap("damage");
+    onMap = true;
+  }
 
-    if (onMap && core.bigmap.v2) {
-        // 检查是否需要重算...
-        if (Math.abs(core.bigmap.posX - core.status.damage.posX) >= core.bigmap.extend - 1
-            || Math.abs(core.bigmap.posY - core.status.damage.posY) >= core.bigmap.extend - 1) {
-            return this.updateDamage();
-        }
+  if (onMap && core.bigmap.v2) {
+    // 检查是否需要重算...
+    if (
+      Math.abs(core.bigmap.posX - core.status.damage.posX) >=
+        core.bigmap.extend - 1 ||
+      Math.abs(core.bigmap.posY - core.status.damage.posY) >=
+        core.bigmap.extend - 1
+    ) {
+      return this.updateDamage();
     }
-    return this._drawDamage_draw(ctx, onMap);
-}
+  }
+  return this._drawDamage_draw(ctx, onMap);
+};
 
 control.prototype._drawDamage_draw = function (ctx, onMap) {
-    if (!core.hasItem('book')) return;
+  if (!core.hasItem("book")) return;
 
-    core.setFont(ctx, "bold 11px Arial");
-    core.setTextAlign(ctx, 'left');
-    core.status.damage.data.forEach(function (one) {
-        var px = one.px, py = one.py;
-        if (onMap && core.bigmap.v2) {
-            px -= core.bigmap.posX * 32;
-            py -= core.bigmap.posY * 32;
-            if (px < -32 * 2 || px > core._PX_ + 32 || py < -32 || py > core._PY_ + 32)
-                return;
-        }
-        core.fillBoldText(ctx, one.text, px, py, one.color);
-    });
+  core.setFont(ctx, "bold 11px Arial");
+  core.setTextAlign(ctx, "left");
+  core.status.damage.data.forEach(function (one) {
+    var px = one.px,
+      py = one.py;
+    if (onMap && core.bigmap.v2) {
+      px -= core.bigmap.posX * 32;
+      py -= core.bigmap.posY * 32;
+      if (
+        px < -32 * 2 ||
+        px > core._PX_ + 32 ||
+        py < -32 ||
+        py > core._PY_ + 32
+      )
+        return;
+    }
+    core.fillBoldText(ctx, one.text, px, py, one.color);
+  });
 
-    core.setTextAlign(ctx, 'center');
-    core.status.damage.extraData.forEach(function (one) {
-        var px = one.px, py = one.py;
-        if (onMap && core.bigmap.v2) {
-            px -= core.bigmap.posX * 32;
-            py -= core.bigmap.posY * 32;
-            if (px < -32 || px > core._PX_ + 32 || py < -32 || py > core._PY_ + 32)
-                return;
-        }
-        var alpha = core.setAlpha(ctx, one.alpha);
-        core.fillBoldText(ctx, one.text, px, py, one.color);
-        core.setAlpha(ctx, alpha);
-    });
-}
+  core.setTextAlign(ctx, "center");
+  core.status.damage.extraData.forEach(function (one) {
+    var px = one.px,
+      py = one.py;
+    if (onMap && core.bigmap.v2) {
+      px -= core.bigmap.posX * 32;
+      py -= core.bigmap.posY * 32;
+      if (px < -32 || px > core._PX_ + 32 || py < -32 || py > core._PY_ + 32)
+        return;
+    }
+    var alpha = core.setAlpha(ctx, one.alpha);
+    core.fillBoldText(ctx, one.text, px, py, one.color);
+    core.setAlpha(ctx, alpha);
+  });
+};
 
 // ------ 录像相关 ------ //
 
 ////// 选择录像文件 //////
 control.prototype.chooseReplayFile = function () {
-    core.readFile(function (obj) {
-        if (obj.name != core.firstData.name) return alert("存档和游戏不一致!");
-        if (!obj.route) return alert("无效的录像!");
-        var _replay = function () {
-            core.startGame(core.flags.startUsingCanvas ? '' : obj.hard || '', obj.seed, core.decodeRoute(obj.route));
-        }
-        if (obj.version && obj.version != core.firstData.version) {
-            core.myconfirm("游戏版本不一致!\n你仍然想播放录像吗?", _replay);
-            return;
-        }
-        _replay();
-    }, null, ".h5route");
-}
+  core.readFile(
+    function (obj) {
+      if (obj.name != core.firstData.name) return alert("存档和游戏不一致!");
+      if (!obj.route) return alert("无效的录像!");
+      var _replay = function () {
+        core.startGame(
+          core.flags.startUsingCanvas ? "" : obj.hard || "",
+          obj.seed,
+          core.decodeRoute(obj.route)
+        );
+      };
+      if (obj.version && obj.version != core.firstData.version) {
+        core.myconfirm("游戏版本不一致!\n你仍然想播放录像吗?", _replay);
+        return;
+      }
+      _replay();
+    },
+    null,
+    ".h5route"
+  );
+};
 
 ////// 开始播放 //////
 control.prototype.startReplay = function (list) {
-    if (!core.isPlaying()) return;
-    core.status.replay.replaying = true;
-    core.status.replay.pausing = true;
-    core.status.replay.failed = false;
-    core.status.replay.speed = 1.0;
-    core.status.replay.toReplay = core.cloneArray(list);
-    core.status.replay.totalList = core.status.route.concat(list);
-    core.status.replay.steps = 0;
-    core.status.replay.save = [];
-    core.createCanvas('replay', 0, core._PY_ - 40, core._PX_, 40, 199);
-    core.setOpacity('replay', 0.6);
-    this._replay_drawProgress();
-    core.updateStatusBar(false, true);
-    core.drawTip("开始播放");
-    this.replay();
-}
+  if (!core.isPlaying()) return;
+  core.status.replay.replaying = true;
+  core.status.replay.pausing = true;
+  core.status.replay.failed = false;
+  core.status.replay.speed = 1.0;
+  core.status.replay.toReplay = core.cloneArray(list);
+  core.status.replay.totalList = core.status.route.concat(list);
+  core.status.replay.steps = 0;
+  core.status.replay.save = [];
+  core.createCanvas("replay", 0, core._PY_ - 40, core._PX_, 40, 199);
+  core.setOpacity("replay", 0.6);
+  this._replay_drawProgress();
+  core.updateStatusBar(false, true);
+  core.drawTip("开始播放");
+  this.replay();
+};
 
 ////// 更改播放状态 //////
 control.prototype.triggerReplay = function () {
-    if (core.status.replay.pausing) this.resumeReplay();
-    else this.pauseReplay();
-}
+  if (core.status.replay.pausing) this.resumeReplay();
+  else this.pauseReplay();
+};
 
 ////// 暂停播放 //////
 control.prototype.pauseReplay = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    core.status.replay.pausing = true;
-    core.updateStatusBar(false, true);
-    core.drawTip("暂停播放");
-}
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  core.status.replay.pausing = true;
+  core.updateStatusBar(false, true);
+  core.drawTip("暂停播放");
+};
 
 ////// 恢复播放 //////
 control.prototype.resumeReplay = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    core.status.replay.pausing = false;
-    core.updateStatusBar(false, true);
-    core.drawTip("恢复播放");
-    core.replay();
-}
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  core.status.replay.pausing = false;
+  core.updateStatusBar(false, true);
+  core.drawTip("恢复播放");
+  core.replay();
+};
 
 ////// 单步播放 //////
 control.prototype.stepReplay = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (!core.status.replay.pausing) {
-        core.playSound('操作失败');
-        return core.drawTip("请先暂停录像");
-    }
-    if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    core.replay(true);
-}
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (!core.status.replay.pausing) {
+    core.playSound("操作失败");
+    return core.drawTip("请先暂停录像");
+  }
+  if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  core.replay(true);
+};
 
 ////// 加速播放 //////
 control.prototype.speedUpReplay = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    var speeds = [0.2, 0.5, 1, 2, 3, 6, 12, 24];
-    for (var i = speeds.length - 2; i >= 0; i--) {
-        if (speeds[i] <= core.status.replay.speed) {
-            core.status.replay.speed = speeds[i + 1];
-            break;
-        }
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  var speeds = [0.2, 0.5, 1, 2, 3, 6, 12, 24];
+  for (var i = speeds.length - 2; i >= 0; i--) {
+    if (speeds[i] <= core.status.replay.speed) {
+      core.status.replay.speed = speeds[i + 1];
+      break;
     }
-    core.drawTip("x" + core.status.replay.speed + "倍");
-}
+  }
+  core.drawTip("x" + core.status.replay.speed + "倍");
+};
 
 ////// 减速播放 //////
 control.prototype.speedDownReplay = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    var speeds = [0.2, 0.5, 1, 2, 3, 6, 12, 24];
-    for (var i = 1; i <= speeds.length; i++) {
-        if (speeds[i] >= core.status.replay.speed) {
-            core.status.replay.speed = speeds[i - 1];
-            break;
-        }
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  var speeds = [0.2, 0.5, 1, 2, 3, 6, 12, 24];
+  for (var i = 1; i <= speeds.length; i++) {
+    if (speeds[i] >= core.status.replay.speed) {
+      core.status.replay.speed = speeds[i - 1];
+      break;
     }
-    core.drawTip("x" + core.status.replay.speed + "倍");
-}
+  }
+  core.drawTip("x" + core.status.replay.speed + "倍");
+};
 
 ////// 设置播放速度 //////
 control.prototype.setReplaySpeed = function (speed) {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    core.status.replay.speed = speed;
-    core.drawTip("x" + core.status.replay.speed + "倍");
-}
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  core.status.replay.speed = speed;
+  core.drawTip("x" + core.status.replay.speed + "倍");
+};
 
 ////// 停止播放 //////
 control.prototype.stopReplay = function (force) {
-    if (!core.isPlaying()) return;
-    if (!core.isReplaying() && !force) return;
-    core.status.replay.toReplay = [];
-    core.status.replay.totalList = [];
-    core.status.replay.replaying = false;
-    core.status.replay.pausing = false;
-    core.status.replay.failed = false;
-    core.status.replay.speed = 1.0;
-    core.status.replay.steps = 0;
-    core.status.replay.save = [];
-    core.deleteCanvas('replay');
-    core.updateStatusBar(false, true);
-    core.drawTip("停止播放并恢复游戏");
-}
+  if (!core.isPlaying()) return;
+  if (!core.isReplaying() && !force) return;
+  core.status.replay.toReplay = [];
+  core.status.replay.totalList = [];
+  core.status.replay.replaying = false;
+  core.status.replay.pausing = false;
+  core.status.replay.failed = false;
+  core.status.replay.speed = 1.0;
+  core.status.replay.steps = 0;
+  core.status.replay.save = [];
+  core.deleteCanvas("replay");
+  core.updateStatusBar(false, true);
+  core.drawTip("停止播放并恢复游戏");
+};
 
 ////// 回退 //////
 control.prototype.rewindReplay = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (!core.status.replay.pausing) {
-        core.playSound('操作失败');
-        return core.drawTip("请先暂停录像");
-    }
-    if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    if (core.status.replay.save.length == 0) {
-        core.playSound('操作失败');
-        return core.drawTip("无法再回到上一个节点");
-    }
-    var save = core.status.replay.save, data = save.pop();
-    core.loadData(data.data, function () {
-        core.removeFlag('__fromLoad__');
-        core.status.replay = {
-            "replaying": true,
-            "pausing": true,
-            "animate": false,
-            "toReplay": data.replay.toReplay,
-            "totalList": data.replay.totalList,
-            "speed": core.status.replay.speed,
-            "steps": data.replay.steps,
-            "save": save
-        }
-        core.createCanvas('replay', 0, core._PY_ - 40, core._PX_, 40, 199);
-        core.setOpacity('replay', 0.6);
-        core.control._replay_drawProgress();
-        core.updateStatusBar(false, true);
-        core.drawTip("成功回退到上一个节点");
-    });
-}
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (!core.status.replay.pausing) {
+    core.playSound("操作失败");
+    return core.drawTip("请先暂停录像");
+  }
+  if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  if (core.status.replay.save.length == 0) {
+    core.playSound("操作失败");
+    return core.drawTip("无法再回到上一个节点");
+  }
+  var save = core.status.replay.save,
+    data = save.pop();
+  core.loadData(data.data, function () {
+    core.removeFlag("__fromLoad__");
+    core.status.replay = {
+      replaying: true,
+      pausing: true,
+      animate: false,
+      toReplay: data.replay.toReplay,
+      totalList: data.replay.totalList,
+      speed: core.status.replay.speed,
+      steps: data.replay.steps,
+      save: save,
+    };
+    core.createCanvas("replay", 0, core._PY_ - 40, core._PX_, 40, 199);
+    core.setOpacity("replay", 0.6);
+    core.control._replay_drawProgress();
+    core.updateStatusBar(false, true);
+    core.drawTip("成功回退到上一个节点");
+  });
+};
 
 ////// 回放时存档 //////
 control.prototype._replay_SL = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (!core.status.replay.pausing) {
-        core.playSound('操作失败');
-        return core.drawTip("请先暂停录像");
-    }
-    if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    if (core.hasFlag('__forbidSave__')) {
-        core.playSound('操作失败');
-        return core.drawTip('当前禁止存档');
-    }
-    this._replay_hideProgress();
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (!core.status.replay.pausing) {
+    core.playSound("操作失败");
+    return core.drawTip("请先暂停录像");
+  }
+  if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  if (core.hasFlag("__forbidSave__")) {
+    core.playSound("操作失败");
+    return core.drawTip("当前禁止存档");
+  }
+  this._replay_hideProgress();
 
-    core.lockControl();
-    core.status.event.id = 'save';
-    var saveIndex = core.saves.saveIndex;
-    var page = parseInt((saveIndex - 1) / 5), offset = saveIndex - 5 * page;
+  core.lockControl();
+  core.status.event.id = "save";
+  var saveIndex = core.saves.saveIndex;
+  var page = parseInt((saveIndex - 1) / 5),
+    offset = saveIndex - 5 * page;
 
-    core.ui._drawSLPanel(10 * page + offset);
-}
+  core.ui._drawSLPanel(10 * page + offset);
+};
 
 ////// 回放时查看怪物手册 //////
 control.prototype._replay_book = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (!core.status.replay.pausing) {
-        core.playSound('操作失败');
-        return core.drawTip("请先暂停录像");
-    }
-    if (core.isMoving() || core.status.replay.animate || (core.status.event.id && core.status.event.id != 'viewMaps')) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    if (!core.hasItem('book')) {
-        core.playSound('操作失败');
-        return core.drawTip('你没有' + core.material.items['book'].name, 'book');
-    }
-    this._replay_hideProgress();
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (!core.status.replay.pausing) {
+    core.playSound("操作失败");
+    return core.drawTip("请先暂停录像");
+  }
+  if (
+    core.isMoving() ||
+    core.status.replay.animate ||
+    (core.status.event.id && core.status.event.id != "viewMaps")
+  ) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  if (!core.hasItem("book")) {
+    core.playSound("操作失败");
+    return core.drawTip("你没有" + core.material.items["book"].name, "book");
+  }
+  this._replay_hideProgress();
 
-    // 从“浏览地图”页面打开
-    if (core.status.event.id == 'viewMaps')
-        core.status.event.ui = core.status.event.data;
+  // 从“浏览地图”页面打开
+  if (core.status.event.id == "viewMaps")
+    core.status.event.ui = core.status.event.data;
 
-    core.lockControl();
-    core.status.event.id = 'book';
-    core.useItem('book', true);
-}
+  core.lockControl();
+  core.status.event.id = "book";
+  core.useItem("book", true);
+};
 
 ////// 回放录像时浏览地图 //////
 control.prototype._replay_viewMap = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (!core.status.replay.pausing) {
-        core.playSound('操作失败');
-        return core.drawTip("请先暂停录像");
-    }
-    if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    this._replay_hideProgress();
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (!core.status.replay.pausing) {
+    core.playSound("操作失败");
+    return core.drawTip("请先暂停录像");
+  }
+  if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  this._replay_hideProgress();
 
-    core.lockControl();
-    core.status.event.id = 'viewMaps';
-    core.ui._drawViewMaps();
-}
+  core.lockControl();
+  core.status.event.id = "viewMaps";
+  core.ui._drawViewMaps();
+};
 
 control.prototype._replay_toolbox = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (!core.status.replay.pausing) {
-        core.playSound('操作失败');
-        return core.drawTip("请先暂停录像");
-    }
-    if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    this._replay_hideProgress();
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (!core.status.replay.pausing) {
+    core.playSound("操作失败");
+    return core.drawTip("请先暂停录像");
+  }
+  if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  this._replay_hideProgress();
 
-    core.lockControl();
-    core.status.event.id = 'toolbox';
-    core.ui._drawToolbox();
-}
+  core.lockControl();
+  core.status.event.id = "toolbox";
+  core.ui._drawToolbox();
+};
 
 control.prototype._replay_equipbox = function () {
-    if (!core.isPlaying() || !core.isReplaying()) return;
-    if (!core.status.replay.pausing) {
-        core.playSound('操作失败');
-        return core.drawTip("请先暂停录像");
-    }
-    if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
-        core.playSound('操作失败');
-        return core.drawTip("请等待当前事件的处理结束");
-    }
-    this._replay_hideProgress();
+  if (!core.isPlaying() || !core.isReplaying()) return;
+  if (!core.status.replay.pausing) {
+    core.playSound("操作失败");
+    return core.drawTip("请先暂停录像");
+  }
+  if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+    core.playSound("操作失败");
+    return core.drawTip("请等待当前事件的处理结束");
+  }
+  this._replay_hideProgress();
 
-    core.lockControl();
-    core.status.event.id = 'equipbox';
-    core.ui._drawEquipbox();
-}
+  core.lockControl();
+  core.status.event.id = "equipbox";
+  core.ui._drawEquipbox();
+};
 
 ////// 是否正在播放录像 //////
 control.prototype.isReplaying = function () {
-    return (core.status.replay || {}).replaying;
-}
+  return (core.status.replay || {}).replaying;
+};
 
 ////// 回放 //////
 control.prototype.replay = function (force) {
-    if (!core.isPlaying() || !core.isReplaying()
-        || core.status.replay.animate || core.status.event.id || core.status.replay.failed) return;
-    if (core.status.replay.pausing && !force) return;
-    this._replay_drawProgress();
-    if (core.status.replay.toReplay.length == 0)
-        return this._replay_finished();
-    this._replay_save();
-    var action = core.status.replay.toReplay.shift();
-    if (this._doReplayAction(action)) return;
-    this._replay_error(action);
-}
+  if (
+    !core.isPlaying() ||
+    !core.isReplaying() ||
+    core.status.replay.animate ||
+    core.status.event.id ||
+    core.status.replay.failed
+  )
+    return;
+  if (core.status.replay.pausing && !force) return;
+  this._replay_drawProgress();
+  if (core.status.replay.toReplay.length == 0) return this._replay_finished();
+  this._replay_save();
+  var action = core.status.replay.toReplay.shift();
+  if (this._doReplayAction(action)) return;
+  this._replay_error(action);
+};
 
 ////// 注册一个录像行为 //////
 // name:自定义名称,可用于注销使用
@@ -1608,1958 +2039,2394 @@ control.prototype.replay = function (force) {
 //       需要接受一个action参数,代表录像回放时的下一个操作
 // func返回true代表成功处理了此录像行为,false代表没有处理此录像行为。
 control.prototype.registerReplayAction = function (name, func) {
-    this.unregisterReplayAction(name);
-    this.replayActions.push({ name: name, func: func });
-}
+  this.unregisterReplayAction(name);
+  this.replayActions.push({ name: name, func: func });
+};
 
 ////// 注销一个录像行为 //////
 control.prototype.unregisterReplayAction = function (name) {
-    this.replayActions = this.replayActions.filter(function (b) { return b.name != name; });
-}
+  this.replayActions = this.replayActions.filter(function (b) {
+    return b.name != name;
+  });
+};
 
 ////// 执行录像行为,会在注册的函数中依次执行直到得到true为止 //////
 control.prototype._doReplayAction = function (action) {
-    for (var i in this.replayActions) {
-        try {
-            if (core.doFunc(this.replayActions[i].func, this, action)) return true;
-        } catch (e) {
-            console.error(e);
-            console.error("ERROR in replayActions[" + this.replayActions[i].name + "]:已自动注销该项。");
-            core.unregisterReplayAction(this.replayActions[i].name);
-        }
+  for (var i in this.replayActions) {
+    try {
+      if (core.doFunc(this.replayActions[i].func, this, action)) return true;
+    } catch (e) {
+      console.error(e);
+      console.error(
+        "ERROR in replayActions[" +
+          this.replayActions[i].name +
+          "]:已自动注销该项。"
+      );
+      core.unregisterReplayAction(this.replayActions[i].name);
     }
-    return false;
-}
+  }
+  return false;
+};
 
 control.prototype._replay_finished = function () {
-    core.status.replay.replaying = false;
-    core.status.replay.failed = false;
-    core.status.event.selection = 0;
-    var str = "录像播放完毕,你想退出播放吗?";
-    if (core.status.route.length != core.status.replay.totalList.length
-        || core.subarray(core.status.route, core.status.replay.totalList) == null) {
-        str = "录像播放完毕,但记录不一致。\n请检查录像播放时的二次记录问题。\n你想退出播放吗?";
+  core.status.replay.replaying = false;
+  core.status.replay.failed = false;
+  core.status.event.selection = 0;
+  var str = "录像播放完毕,你想退出播放吗?";
+  if (
+    core.status.route.length != core.status.replay.totalList.length ||
+    core.subarray(core.status.route, core.status.replay.totalList) == null
+  ) {
+    str =
+      "录像播放完毕,但记录不一致。\n请检查录像播放时的二次记录问题。\n你想退出播放吗?";
+  }
+  core.ui.drawConfirmBox(
+    str,
+    function () {
+      core.ui.closePanel();
+      core.stopReplay(true);
+    },
+    function () {
+      core.status.replay.replaying = true;
+      core.ui.closePanel();
+      core.pauseReplay();
     }
-    core.ui.drawConfirmBox(str, function () {
-        core.ui.closePanel();
-        core.stopReplay(true);
-    }, function () {
-        core.status.replay.replaying = true;
-        core.ui.closePanel();
-        core.pauseReplay();
-    });
-}
+  );
+};
 
 control.prototype._replay_save = function () {
-    core.status.replay.steps++;
-    if (core.status.replay.steps % 40 == 1) {
-        if (core.status.replay.save.length == 30)
-            core.status.replay.save.shift();
-        core.status.replay.save.push({
-            "data": core.saveData(), "replay": {
-                "totalList": core.cloneArray(core.status.replay.totalList),
-                "toReplay": core.cloneArray(core.status.replay.toReplay),
-                "steps": core.status.replay.steps
-            }
-        });
-    }
-}
+  core.status.replay.steps++;
+  if (core.status.replay.steps % 40 == 1) {
+    if (core.status.replay.save.length == 30) core.status.replay.save.shift();
+    core.status.replay.save.push({
+      data: core.saveData(),
+      replay: {
+        totalList: core.cloneArray(core.status.replay.totalList),
+        toReplay: core.cloneArray(core.status.replay.toReplay),
+        steps: core.status.replay.steps,
+      },
+    });
+  }
+};
 
 control.prototype._replay_error = function (action, callback) {
-    core.ui.closePanel();
-    core.status.replay.replaying = false;
-    core.status.replay.failed = true;
-    var len = core.status.replay.toReplay.length;
-    var prevList = core.status.replay.totalList.slice(-len - 11, -len - 1);
-    var nextList = core.status.replay.toReplay.slice(0, 10);
-    console.log("录像文件出错,当前操作:" + action);
-    console.log("之前的10个操作是:\n" + prevList.toString());
-    console.log("接下来10个操作是:\n" + nextList.toString());
-    core.ui.drawConfirmBox("录像文件出错,你想回到上个节点吗?", function () {
-        core.status.replay.failed = false;
-        core.ui.closePanel();
-        if (core.status.replay.save.length > 0) {
-            core.status.replay.replaying = true;
-            core.status.replay.pausing = true;
-            core.rewindReplay();
-        }
-        else {
-            core.playSound('操作失败');
-            core.stopReplay(true);
-            core.drawTip("无法回到上一个节点");
-            if (callback) callback();
-        }
-    }, function () {
-        core.status.replay.failed = false;
-        core.ui.closePanel();
+  core.ui.closePanel();
+  core.status.replay.replaying = false;
+  core.status.replay.failed = true;
+  var len = core.status.replay.toReplay.length;
+  var prevList = core.status.replay.totalList.slice(-len - 11, -len - 1);
+  var nextList = core.status.replay.toReplay.slice(0, 10);
+  console.log("录像文件出错,当前操作:" + action);
+  console.log("之前的10个操作是:\n" + prevList.toString());
+  console.log("接下来10个操作是:\n" + nextList.toString());
+  core.ui.drawConfirmBox(
+    "录像文件出错,你想回到上个节点吗?",
+    function () {
+      core.status.replay.failed = false;
+      core.ui.closePanel();
+      if (core.status.replay.save.length > 0) {
+        core.status.replay.replaying = true;
+        core.status.replay.pausing = true;
+        core.rewindReplay();
+      } else {
+        core.playSound("操作失败");
         core.stopReplay(true);
+        core.drawTip("无法回到上一个节点");
         if (callback) callback();
-    });
-}
+      }
+    },
+    function () {
+      core.status.replay.failed = false;
+      core.ui.closePanel();
+      core.stopReplay(true);
+      if (callback) callback();
+    }
+  );
+};
 
 control.prototype._replay_hideProgress = function () {
-    if (core.dymCanvas.replay) core.dymCanvas.replay.canvas.style.display = 'none';
-}
+  if (core.dymCanvas.replay)
+    core.dymCanvas.replay.canvas.style.display = "none";
+};
 
 control.prototype._replay_drawProgress = function () {
-    if (!core.dymCanvas.replay) return;
-    if (core.dymCanvas.replay.canvas.style.display == 'none') core.dymCanvas.replay.canvas.style.display = 'block';
-    var total = core.status.replay.totalList.length, left = total - core.status.replay.toReplay.length;
-    var content = '播放进度:' + left + ' / ' + total + '(' + (left / total * 100).toFixed(2) + '%)';
-    var width = 26 + core.calWidth('replay', content, "16px Arial");
-    core.clearMap('replay');
-    core.fillRect('replay', 0, 0, width, 40, '#000000');
-    core.fillText('replay', content, 16, 27, '#FFFFFF');
-}
+  if (!core.dymCanvas.replay) return;
+  if (core.dymCanvas.replay.canvas.style.display == "none")
+    core.dymCanvas.replay.canvas.style.display = "block";
+  var total = core.status.replay.totalList.length,
+    left = total - core.status.replay.toReplay.length;
+  var content =
+    "播放进度:" +
+    left +
+    " / " +
+    total +
+    "(" +
+    ((left / total) * 100).toFixed(2) +
+    "%)";
+  var width = 26 + core.calWidth("replay", content, "16px Arial");
+  core.clearMap("replay");
+  core.fillRect("replay", 0, 0, width, 40, "#000000");
+  core.fillText("replay", content, 16, 27, "#FFFFFF");
+};
 
 control.prototype.__replay_getTimeout = function () {
-    if (core.status.replay.speed == 24) return 0;
-    return 750 / Math.max(1, core.status.replay.speed);
-}
+  if (core.status.replay.speed == 24) return 0;
+  return 750 / Math.max(1, core.status.replay.speed);
+};
 
 control.prototype._replayAction_move = function (action) {
-    if (["up", "down", "left", "right"].indexOf(action) < 0) return false;
-    core.moveHero(action, core.replay);
-    return true;
-}
+  if (["up", "down", "left", "right"].indexOf(action) < 0) return false;
+  core.moveHero(action, core.replay);
+  return true;
+};
 
 control.prototype._replayAction_item = function (action) {
-    if (action.indexOf("item:") != 0) return false;
-    var itemId = action.substring(5);
-    if (!core.canUseItem(itemId)) return false;
-    if (core.material.items[itemId].hideInReplay || core.status.replay.speed == 24) {
-        core.useItem(itemId, false, core.replay);
-        return true;
-    }
-    var tools = core.getToolboxItems('tools'),
-        constants = core.getToolboxItems('constants');
-    var index, per = core._WIDTH_ - 1;
-    if ((index = tools.indexOf(itemId)) >= 0) {
-        core.status.event.data = { "toolsPage": Math.floor(index / per) + 1, "constantsPage": 1 };
-        index = index % per;
-    }
-    else if ((index = constants.indexOf(itemId)) >= 0) {
-        core.status.event.data = { "toolsPage": 1, "constantsPage": Math.floor(index / per) + 1 };
-        index = index % per + per;
-    }
-    if (index < 0) return false;
-    core.ui._drawToolbox(index);
-    setTimeout(function () {
-        core.ui.closePanel();
-        core.useItem(itemId, false, core.replay);
-    }, core.control.__replay_getTimeout());
+  if (action.indexOf("item:") != 0) return false;
+  var itemId = action.substring(5);
+  if (!core.canUseItem(itemId)) return false;
+  if (
+    core.material.items[itemId].hideInReplay ||
+    core.status.replay.speed == 24
+  ) {
+    core.useItem(itemId, false, core.replay);
     return true;
-}
+  }
+  var tools = core.getToolboxItems("tools"),
+    constants = core.getToolboxItems("constants");
+  var index,
+    per = core._WIDTH_ - 1;
+  if ((index = tools.indexOf(itemId)) >= 0) {
+    core.status.event.data = {
+      toolsPage: Math.floor(index / per) + 1,
+      constantsPage: 1,
+    };
+    index = index % per;
+  } else if ((index = constants.indexOf(itemId)) >= 0) {
+    core.status.event.data = {
+      toolsPage: 1,
+      constantsPage: Math.floor(index / per) + 1,
+    };
+    index = (index % per) + per;
+  }
+  if (index < 0) return false;
+  core.ui._drawToolbox(index);
+  setTimeout(function () {
+    core.ui.closePanel();
+    core.useItem(itemId, false, core.replay);
+  }, core.control.__replay_getTimeout());
+  return true;
+};
 
 control.prototype._replayAction_equip = function (action) {
-    if (action.indexOf("equip:") != 0) return false;
-    var equipId = action.substring(6);
-    var ownEquipment = core.getToolboxItems('equips');
-    var index = ownEquipment.indexOf(equipId), per = core._WIDTH_ - 1;
-    if (index < 0) {
-        core.removeFlag('__doNotCheckAutoEvents__');
-        return false;
-    }
+  if (action.indexOf("equip:") != 0) return false;
+  var equipId = action.substring(6);
+  var ownEquipment = core.getToolboxItems("equips");
+  var index = ownEquipment.indexOf(equipId),
+    per = core._WIDTH_ - 1;
+  if (index < 0) {
+    core.removeFlag("__doNotCheckAutoEvents__");
+    return false;
+  }
 
-    var cb = function () {
-        var next = core.status.replay.toReplay[0] || "";
-        if (!next.startsWith('equip:') && !next.startsWith('unEquip:')) {
-            core.removeFlag('__doNotCheckAutoEvents__');
-            core.checkAutoEvents();
-        }
-        core.replay();
+  var cb = function () {
+    var next = core.status.replay.toReplay[0] || "";
+    if (!next.startsWith("equip:") && !next.startsWith("unEquip:")) {
+      core.removeFlag("__doNotCheckAutoEvents__");
+      core.checkAutoEvents();
     }
-    core.setFlag('__doNotCheckAutoEvents__', true);
+    core.replay();
+  };
+  core.setFlag("__doNotCheckAutoEvents__", true);
 
-    core.status.route.push(action);
-    if (core.material.items[equipId].hideInReplay || core.status.replay.speed == 24) {
-        core.loadEquip(equipId, cb);
-        return true;
-    }
-    core.status.event.data = { "page": Math.floor(index / per) + 1, "selectId": null };
-    index = index % per + per;
-    core.ui._drawEquipbox(index);
-    setTimeout(function () {
-        core.ui.closePanel();
-        core.loadEquip(equipId, cb);
-    }, core.control.__replay_getTimeout());
+  core.status.route.push(action);
+  if (
+    core.material.items[equipId].hideInReplay ||
+    core.status.replay.speed == 24
+  ) {
+    core.loadEquip(equipId, cb);
     return true;
-}
+  }
+  core.status.event.data = {
+    page: Math.floor(index / per) + 1,
+    selectId: null,
+  };
+  index = (index % per) + per;
+  core.ui._drawEquipbox(index);
+  setTimeout(function () {
+    core.ui.closePanel();
+    core.loadEquip(equipId, cb);
+  }, core.control.__replay_getTimeout());
+  return true;
+};
 
 control.prototype._replayAction_unEquip = function (action) {
-    if (action.indexOf("unEquip:") != 0) return false;
-    var equipType = parseInt(action.substring(8));
-    if (!core.isset(equipType)) {
-        core.removeFlag('__doNotCheckAutoEvents__');
-        return false;
-    }
+  if (action.indexOf("unEquip:") != 0) return false;
+  var equipType = parseInt(action.substring(8));
+  if (!core.isset(equipType)) {
+    core.removeFlag("__doNotCheckAutoEvents__");
+    return false;
+  }
 
-    var cb = function () {
-        var next = core.status.replay.toReplay[0] || "";
-        if (!next.startsWith('equip:') && !next.startsWith('unEquip:')) {
-            core.removeFlag('__doNotCheckAutoEvents__');
-            core.checkAutoEvents();
-        }
-        core.replay();
+  var cb = function () {
+    var next = core.status.replay.toReplay[0] || "";
+    if (!next.startsWith("equip:") && !next.startsWith("unEquip:")) {
+      core.removeFlag("__doNotCheckAutoEvents__");
+      core.checkAutoEvents();
     }
-    core.setFlag('__doNotCheckAutoEvents__', true);
+    core.replay();
+  };
+  core.setFlag("__doNotCheckAutoEvents__", true);
 
-    core.ui._drawEquipbox(equipType);
-    core.status.route.push(action);
-    if (core.status.replay.speed == 24) {
-        core.unloadEquip(equipType, cb);
-        return true;
-    }
-    setTimeout(function () {
-        core.ui.closePanel();
-        core.unloadEquip(equipType, cb);
-    }, core.control.__replay_getTimeout());
+  core.ui._drawEquipbox(equipType);
+  core.status.route.push(action);
+  if (core.status.replay.speed == 24) {
+    core.unloadEquip(equipType, cb);
     return true;
-}
+  }
+  setTimeout(function () {
+    core.ui.closePanel();
+    core.unloadEquip(equipType, cb);
+  }, core.control.__replay_getTimeout());
+  return true;
+};
 
 control.prototype._replayAction_saveEquip = function (action) {
-    if (action.indexOf('saveEquip:') != 0) return false;
-    core.quickSaveEquip(parseInt(action.substring(10)));
-    core.replay();
-    return true;
-}
+  if (action.indexOf("saveEquip:") != 0) return false;
+  core.quickSaveEquip(parseInt(action.substring(10)));
+  core.replay();
+  return true;
+};
 
 control.prototype._replayAction_loadEquip = function (action) {
-    if (action.indexOf('loadEquip:') != 0) return false;
-    core.quickLoadEquip(parseInt(action.substring(10)));
-    core.replay();
-    return true;
-}
+  if (action.indexOf("loadEquip:") != 0) return false;
+  core.quickLoadEquip(parseInt(action.substring(10)));
+  core.replay();
+  return true;
+};
 
 control.prototype._replayAction_fly = function (action) {
-    if (action.indexOf("fly:") != 0) return false;
-    var floorId = action.substring(4);
-    var toIndex = core.floorIds.indexOf(floorId);
-    if (!core.canUseItem('fly') || (core.flags.flyNearStair && !core.nearStair())) return false;
-    core.ui.drawFly(toIndex);
-    if (core.status.replay.speed == 24) {
-        if (!core.flyTo(floorId, core.replay))
-            core.control._replay_error(action);
-        return true;
-    }
-    setTimeout(function () {
-        if (!core.flyTo(floorId, core.replay))
-            core.control._replay_error(action);
-    }, core.control.__replay_getTimeout());
+  if (action.indexOf("fly:") != 0) return false;
+  var floorId = action.substring(4);
+  var toIndex = core.floorIds.indexOf(floorId);
+  if (
+    !core.canUseItem("fly") ||
+    (core.flags.flyNearStair && !core.nearStair() && !flags.canMoveFloor)
+  )
+    return false;
+  core.ui.drawFly(toIndex);
+  if (core.status.replay.speed == 24) {
+    if (!core.flyTo(floorId, core.replay)) core.control._replay_error(action);
     return true;
-}
+  }
+  setTimeout(function () {
+    if (!core.flyTo(floorId, core.replay)) core.control._replay_error(action);
+  }, core.control.__replay_getTimeout());
+  return true;
+};
 
 control.prototype._replayAction_shop = function (action) {
-    if (action.indexOf("shop:") != 0) return false;
-    var shopId = action.substring(5);
-    if (core.canUseQuickShop(shopId) != null || !core.canOpenShop(shopId)) {
-        this._replay_error(shopId);
-        return true;
-    }
-    core.openShop(shopId, false);
-    core.replay();
+  if (action.indexOf("shop:") != 0) return false;
+  var shopId = action.substring(5);
+  if (core.canUseQuickShop(shopId) != null || !core.canOpenShop(shopId)) {
+    this._replay_error(shopId);
     return true;
-}
+  }
+  core.openShop(shopId, false);
+  core.replay();
+  return true;
+};
 
 control.prototype._replayAction_turn = function (action) {
-    if (action != 'turn' && action.indexOf('turn:') != 0) return false;
-    if (action == 'turn') core.turnHero();
-    else core.turnHero(action.substring(5));
-    core.replay();
-    return true;
-}
+  if (action != "turn" && action.indexOf("turn:") != 0) return false;
+  if (action == "turn") core.turnHero();
+  else core.turnHero(action.substring(5));
+  core.replay();
+  return true;
+};
 
 control.prototype._replayAction_getNext = function (action) {
-    if (action != "getNext") return false;
-    core.getNextItem();
-    core.replay();
-    return true;
-}
+  if (action != "getNext") return false;
+  core.getNextItem();
+  core.replay();
+  return true;
+};
 
 control.prototype._replayAction_moveDirectly = function (action) {
-    if (action.indexOf("move:") != 0) return false;
-    // 忽略连续的瞬移事件;如果大地图某一边超过计算范围则不合并
-    if (!core.hasFlag('poison') && core.status.thisMap.width < 2 * core.bigmap.extend + core._WIDTH_
-        && core.status.thisMap.height < 2 * core.bigmap.extend + core._HEIGHT_) {
-        while (core.status.replay.toReplay.length > 0 &&
-            core.status.replay.toReplay[0].indexOf('move:') == 0) {
-            core.status.route.push(action);
-            action = core.status.replay.toReplay.shift();
-        }
+  if (action.indexOf("move:") != 0) return false;
+  // 忽略连续的瞬移事件;如果大地图某一边超过计算范围则不合并
+  if (
+    !core.hasFlag("poison") &&
+    core.status.thisMap.width < 2 * core.bigmap.extend + core._WIDTH_ &&
+    core.status.thisMap.height < 2 * core.bigmap.extend + core._HEIGHT_
+  ) {
+    while (
+      core.status.replay.toReplay.length > 0 &&
+      core.status.replay.toReplay[0].indexOf("move:") == 0
+    ) {
+      core.status.route.push(action);
+      action = core.status.replay.toReplay.shift();
     }
+  }
 
-    var pos = action.substring(5).split(":");
-    var x = parseInt(pos[0]), y = parseInt(pos[1]);
-    var nowx = core.getHeroLoc('x'), nowy = core.getHeroLoc('y');
-    var ignoreSteps = core.canMoveDirectly(x, y);
-    if (!core.moveDirectly(x, y, ignoreSteps)) return false;
-    if (core.status.replay.speed == 24) {
-        core.replay();
-        return true;
-    }
-
-    core.ui.drawArrow('ui', 32 * nowx + 16 - core.bigmap.offsetX, 32 * nowy + 16 - core.bigmap.offsetY,
-        32 * x + 16 - core.bigmap.offsetX, 32 * y + 16 - core.bigmap.offsetY, '#FF0000', 3);
-    var timeout = this.__replay_getTimeout();
-    if (ignoreSteps < 10) timeout = timeout * ignoreSteps / 10;
-    setTimeout(function () {
-        core.clearMap('ui');
-        core.replay();
-    }, timeout);
+  var pos = action.substring(5).split(":");
+  var x = parseInt(pos[0]),
+    y = parseInt(pos[1]);
+  var nowx = core.getHeroLoc("x"),
+    nowy = core.getHeroLoc("y");
+  var ignoreSteps = core.canMoveDirectly(x, y);
+  if (!core.moveDirectly(x, y, ignoreSteps)) return false;
+  if (core.status.replay.speed == 24) {
+    core.replay();
     return true;
-}
+  }
+
+  core.ui.drawArrow(
+    "ui",
+    32 * nowx + 16 - core.bigmap.offsetX,
+    32 * nowy + 16 - core.bigmap.offsetY,
+    32 * x + 16 - core.bigmap.offsetX,
+    32 * y + 16 - core.bigmap.offsetY,
+    "#FF0000",
+    3
+  );
+  var timeout = this.__replay_getTimeout();
+  if (ignoreSteps < 10) timeout = (timeout * ignoreSteps) / 10;
+  setTimeout(function () {
+    core.clearMap("ui");
+    core.replay();
+  }, timeout);
+  return true;
+};
 
 control.prototype._replayAction_key = function (action) {
-    if (action.indexOf("key:") != 0) return false;
-    core.actions.keyUp(parseInt(action.substring(4)), false, true);
-    core.replay();
-    return true;
-}
+  if (action.indexOf("key:") != 0) return false;
+  core.actions.keyUp(parseInt(action.substring(4)), false, true);
+  core.replay();
+  return true;
+};
 
 control.prototype._replayAction_click = function (action) {
-    if (action.indexOf("click:") != 0) return false;
-    var p = action.split(":");
-    if (p.length != 4) return false;
-    core.actions.doRegisteredAction("onStatusBarClick", parseInt(p[2]), parseInt(p[3]), parseInt(p[1]));
-    core.replay();
-    return true;
-}
+  if (action.indexOf("click:") != 0) return false;
+  var p = action.split(":");
+  if (p.length != 4) return false;
+  core.actions.doRegisteredAction(
+    "onStatusBarClick",
+    parseInt(p[2]),
+    parseInt(p[3]),
+    parseInt(p[1])
+  );
+  core.replay();
+  return true;
+};
 
 control.prototype._replayAction_ignoreInput = function (action) {
-    if (action.indexOf('input:') == 0 || action.indexOf('input2:') == 0 || action.indexOf('choices:') == 0 || action.indexOf('random:') == 0) {
-        console.warn('警告!录像播放中出现了未知的 ' + action + '!');
-        core.replay();
-        return true;
-    }
-    return false;
-}
-
-control.prototype._replayAction_no = function (action) {
-    if (action != 'no') return false;
-    core.status.route.push(action);
+  if (
+    action.indexOf("input:") == 0 ||
+    action.indexOf("input2:") == 0 ||
+    action.indexOf("choices:") == 0 ||
+    action.indexOf("random:") == 0
+  ) {
+    console.warn("警告!录像播放中出现了未知的 " + action + "!");
     core.replay();
     return true;
-}
+  }
+  return false;
+};
+
+control.prototype._replayAction_no = function (action) {
+  if (action != "no") return false;
+  core.status.route.push(action);
+  core.replay();
+  return true;
+};
 
 // ------ 存读档相关 ------ //
 
 ////// 自动存档 //////
 control.prototype.autosave = function (removeLast) {
-    if (core.hasFlag('__forbidSave__')) return;
-    var x = null;
-    if (removeLast) {
-        x = core.status.route.pop();
-        core.status.route.push("turn:" + core.getHeroLoc('direction'));
+  if (core.hasFlag("__forbidSave__")) return;
+  var x = null;
+  if (removeLast) {
+    x = core.status.route.pop();
+    core.status.route.push("turn:" + core.getHeroLoc("direction"));
+  }
+  if (core.status.event.id == "action" && !removeLast)
+    // 事件中自动存档,读档后是否回到事件触发前
+    core.setFlag("__events__", core.clone(core.status.event.data));
+  if (core.saves.autosave.data == null) {
+    core.saves.autosave.data = [];
+  }
+  core.saves.autosave.data.splice(core.saves.autosave.now, 0, core.saveData());
+  core.saves.autosave.now += 1;
+  if (core.saves.autosave.data.length > core.saves.autosave.max) {
+    if (core.saves.autosave.now < core.saves.autosave.max / 2)
+      core.saves.autosave.data.pop();
+    else {
+      core.saves.autosave.data.shift();
+      core.saves.autosave.now = core.saves.autosave.now - 1;
     }
-    if (core.status.event.id == 'action' && !removeLast) // 事件中自动存档,读档后是否回到事件触发前
-        core.setFlag("__events__", core.clone(core.status.event.data));
-    if (core.saves.autosave.data == null) {
-        core.saves.autosave.data = [];
-    }
-    core.saves.autosave.data.splice(core.saves.autosave.now, 0, core.saveData());
-    core.saves.autosave.now += 1;
-    if (core.saves.autosave.data.length > core.saves.autosave.max) {
-        if (core.saves.autosave.now < core.saves.autosave.max / 2)
-            core.saves.autosave.data.pop();
-        else {
-            core.saves.autosave.data.shift();
-            core.saves.autosave.now = core.saves.autosave.now - 1;
-        }
-    }
-    core.saves.autosave.updated = true;
-    core.saves.ids[0] = true;
-    core.removeFlag("__events__");
-    if (removeLast) {
-        core.status.route.pop();
-        if (x) core.status.route.push(x);
-    }
-}
+  }
+  core.saves.autosave.updated = true;
+  core.saves.ids[0] = true;
+  core.removeFlag("__events__");
+  if (removeLast) {
+    core.status.route.pop();
+    if (x) core.status.route.push(x);
+  }
+};
 
 /////// 实际进行自动存档 //////
 control.prototype.checkAutosave = function () {
-    if (!core.animateFrame || !core.saves || !core.saves.autosave) return;
-    core.setLocalStorage('totalTime', core.animateFrame.totalTime);
-    var autosave = core.saves.autosave;
-    if (autosave.data == null || !autosave.updated || !autosave.storage) return;
-    autosave.updated = false;
-    if (autosave.data.length >= 1) {
-        core.setLocalForage("autoSave", autosave.data[autosave.now - 1]);
-    }
-}
+  if (!core.animateFrame || !core.saves || !core.saves.autosave) return;
+  core.setLocalStorage("totalTime", core.animateFrame.totalTime);
+  var autosave = core.saves.autosave;
+  if (autosave.data == null || !autosave.updated || !autosave.storage) return;
+  autosave.updated = false;
+  if (autosave.data.length >= 1) {
+    core.setLocalForage("autoSave", autosave.data[autosave.now - 1]);
+  }
+};
 
 ////// 实际进行存读档事件 //////
 control.prototype.doSL = function (id, type) {
-    switch (type) {
-        case 'save': this._doSL_save(id); break;
-        case 'load': this._doSL_load(id, this._doSL_load_afterGet); break;
-        case 'reload': this._doSL_reload(id, this._doSL_load_afterGet); break;
-        case 'replayLoad': this._doSL_load(id, this._doSL_replayLoad_afterGet); break;
-        case 'replayRemain': this._doSL_load(id, this._doSL_replayRemain_afterGet); break;
-        case 'replaySince': this._doSL_load(id, this._doSL_replaySince_afterGet); break;
-    }
-}
+  switch (type) {
+    case "save":
+      this._doSL_save(id);
+      break;
+    case "load":
+      this._doSL_load(id, this._doSL_load_afterGet);
+      break;
+    case "reload":
+      this._doSL_reload(id, this._doSL_load_afterGet);
+      break;
+    case "replayLoad":
+      this._doSL_load(id, this._doSL_replayLoad_afterGet);
+      break;
+    case "replayRemain":
+      this._doSL_load(id, this._doSL_replayRemain_afterGet);
+      break;
+    case "replaySince":
+      this._doSL_load(id, this._doSL_replaySince_afterGet);
+      break;
+  }
+};
 
 control.prototype._doSL_save = function (id) {
-    if (id == 'autoSave') {
-        core.playSound('操作失败');
-        return core.drawTip('不能覆盖自动存档!');
+  if (id == "autoSave") {
+    core.playSound("操作失败");
+    return core.drawTip("不能覆盖自动存档!");
+  }
+  // 在事件中的存档
+  if (core.status.event.interval != null)
+    core.setFlag("__events__", core.status.event.interval);
+  var data = core.saveData();
+  if (core.isReplaying() && core.status.replay.toReplay.length > 0) {
+    data.__toReplay__ = core.encodeRoute(core.status.replay.toReplay);
+  }
+  core.setLocalForage(
+    "save" + id,
+    data,
+    function () {
+      core.saves.saveIndex = id;
+      core.setLocalStorage("saveIndex", core.saves.saveIndex);
+      // 恢复事件
+      if (!core.events.recoverEvents(core.status.event.interval))
+        core.ui.closePanel();
+      core.playSound("存档");
+      core.drawTip("存档成功!");
+    },
+    function (err) {
+      console.error(err);
+      alert("存档失败,错误信息:\n" + err);
     }
-    // 在事件中的存档
-    if (core.status.event.interval != null)
-        core.setFlag("__events__", core.status.event.interval);
-    var data = core.saveData();
-    if (core.isReplaying() && core.status.replay.toReplay.length > 0) {
-        data.__toReplay__ = core.encodeRoute(core.status.replay.toReplay);
-    }
-    core.setLocalForage("save" + id, data, function () {
-        core.saves.saveIndex = id;
-        core.setLocalStorage('saveIndex', core.saves.saveIndex);
-        // 恢复事件
-        if (!core.events.recoverEvents(core.status.event.interval))
-            core.ui.closePanel();
-        core.playSound('存档');
-        core.drawTip('存档成功!');
-    }, function (err) {
-        console.error(err);
-        alert("存档失败,错误信息:\n" + err);
-    });
-    core.removeFlag("__events__");
-    return;
-}
+  );
+  core.removeFlag("__events__");
+  return;
+};
 
 control.prototype._doSL_load = function (id, callback) {
-    if (id == 'autoSave' && core.saves.autosave.data != null) {
-        core.saves.autosave.now -= 1;
-        var data = core.saves.autosave.data.splice(core.saves.autosave.now, 1)[0];
-        if (core.isPlaying() && !core.status.gameOver) {
-            core.control.autosave(0);
-            core.saves.autosave.now -= 1;
-        }
-        if (core.saves.autosave.now == 0) {
-            core.saves.autosave.data.unshift(core.clone(data));
-            core.saves.autosave.now += 1;
+  if (id == "autoSave" && core.saves.autosave.data != null) {
+    core.saves.autosave.now -= 1;
+    var data = core.saves.autosave.data.splice(core.saves.autosave.now, 1)[0];
+    if (core.isPlaying() && !core.status.gameOver) {
+      core.control.autosave(0);
+      core.saves.autosave.now -= 1;
+    }
+    if (core.saves.autosave.now == 0) {
+      core.saves.autosave.data.unshift(core.clone(data));
+      core.saves.autosave.now += 1;
+    }
+    callback(id, data);
+  } else {
+    core.getLocalForage(
+      id == "autoSave" ? id : "save" + id,
+      null,
+      function (data) {
+        if (id == "autoSave" && data != null) {
+          core.saves.autosave.data = data;
+          if (!(core.saves.autosave.data instanceof Array)) {
+            core.saves.autosave.data = [core.saves.autosave.data];
+          }
+          core.saves.autosave.now = core.saves.autosave.data.length;
+          return core.control._doSL_load(id, callback);
         }
         callback(id, data);
-    }
-    else {
-        core.getLocalForage(id == 'autoSave' ? id : "save" + id, null, function (data) {
-            if (id == 'autoSave' && data != null) {
-                core.saves.autosave.data = data;
-                if (!(core.saves.autosave.data instanceof Array)) {
-                    core.saves.autosave.data = [core.saves.autosave.data];
-                }
-                core.saves.autosave.now = core.saves.autosave.data.length;
-                return core.control._doSL_load(id, callback);
-            }
-            callback(id, data);
-        }, function (err) {
-            console.error(err);
-            alert("无效的存档");
-        })
-    }
-    return;
-}
+      },
+      function (err) {
+        console.error(err);
+        alert("无效的存档");
+      }
+    );
+  }
+  return;
+};
 
 control.prototype._doSL_reload = function (id, callback) {
-    if (core.saves.autosave.data != null && core.saves.autosave.now < core.saves.autosave.data.length) {
-        var data = core.saves.autosave.data.splice(core.saves.autosave.now, 1)[0];
-        core.control.autosave(false);
-        callback(id, data);
-    }
-    return;
-}
+  if (
+    core.saves.autosave.data != null &&
+    core.saves.autosave.now < core.saves.autosave.data.length
+  ) {
+    var data = core.saves.autosave.data.splice(core.saves.autosave.now, 1)[0];
+    core.control.autosave(false);
+    callback(id, data);
+  }
+  return;
+};
 
 control.prototype._doSL_load_afterGet = function (id, data) {
-    if (!data) return alert("无效的存档");
-    var _replay = function () {
-        core.startGame(data.hard, data.hero.flags.__seed__, core.decodeRoute(data.route));
-    };
-    if (data.version != core.firstData.version) {
-        core.myconfirm("存档版本不匹配!\n你想回放此存档的录像吗?\n可以随时停止录像播放以继续游戏。", _replay);
-        return;
+  if (!data) return alert("无效的存档");
+  var _replay = function () {
+    core.startGame(
+      data.hard,
+      data.hero.flags.__seed__,
+      core.decodeRoute(data.route)
+    );
+  };
+  if (data.version != core.firstData.version) {
+    core.myconfirm(
+      "存档版本不匹配!\n你想回放此存档的录像吗?\n可以随时停止录像播放以继续游戏。",
+      _replay
+    );
+    return;
+  }
+  if (data.hero.flags.__events__ && data.guid != core.getGuid()) {
+    core.myconfirm("此存档可能存在风险,你想要播放录像么?", _replay);
+    return;
+  }
+  core.ui.closePanel();
+  core.loadData(data, function () {
+    core.removeFlag("__fromLoad__");
+    core.drawTip("读档成功");
+    if (id != "autoSave") {
+      core.saves.saveIndex = id;
+      core.setLocalStorage("saveIndex", core.saves.saveIndex);
     }
-    if (data.hero.flags.__events__ && data.guid != core.getGuid()) {
-        core.myconfirm("此存档可能存在风险,你想要播放录像么?", _replay);
-        return;
-    }
-    core.ui.closePanel();
-    core.loadData(data, function () {
-        core.removeFlag('__fromLoad__');
-        core.drawTip("读档成功");
-        if (id != "autoSave") {
-            core.saves.saveIndex = id;
-            core.setLocalStorage('saveIndex', core.saves.saveIndex);
-        }
-    });
-}
+  });
+};
 
 control.prototype._doSL_replayLoad_afterGet = function (id, data) {
-    if (!data) {
-        core.playSound('操作失败');
-        return core.drawTip("无效的存档");
-    }
-    if (data.version != core.firstData.version) {
-        core.playSound('操作失败');
-        return core.drawTip("存档版本不匹配");
-    }
-    if (data.hero.flags.__events__ && data.guid != core.getGuid()) {
-        core.playSound('操作失败');
-        return core.drawTip("此存档可能存在风险,无法读档");
-    }
-    var route = core.subarray(core.status.route, core.decodeRoute(data.route));
-    if (route == null) {
-        core.playSound('操作失败');
-        return core.drawTip("无法从此存档回放录像");
-    }
-    core.loadData(data, function () {
-        core.removeFlag('__fromLoad__');
-        core.startReplay(route);
-        core.drawTip("回退到存档节点");
-    });
-}
+  if (!data) {
+    core.playSound("操作失败");
+    return core.drawTip("无效的存档");
+  }
+  if (data.version != core.firstData.version) {
+    core.playSound("操作失败");
+    return core.drawTip("存档版本不匹配");
+  }
+  if (data.hero.flags.__events__ && data.guid != core.getGuid()) {
+    core.playSound("操作失败");
+    return core.drawTip("此存档可能存在风险,无法读档");
+  }
+  var route = core.subarray(core.status.route, core.decodeRoute(data.route));
+  if (route == null) {
+    core.playSound("操作失败");
+    return core.drawTip("无法从此存档回放录像");
+  }
+  core.loadData(data, function () {
+    core.removeFlag("__fromLoad__");
+    core.startReplay(route);
+    core.drawTip("回退到存档节点");
+  });
+};
 
 control.prototype._doSL_replayRemain_afterGet = function (id, data) {
-    if (!data) {
-        core.playSound('操作失败');
-        return core.drawTip("无效的存档");
-    }
-    var route = core.decodeRoute(data.route);
-    if (core.status.tempRoute) {
-        var remainRoute = core.subarray(route, core.status.tempRoute);
-        if (remainRoute == null)
-            return alert("无法接续播放录像!\n该存档必须是前一个选择的存档的后续内容。");
-        delete core.status.tempRoute;
-        core.ui.closePanel();
-        core.startReplay(remainRoute);
-        core.drawTip("接续播放录像");
-        return;
-    }
-    else if (data.floorId != core.status.floorId || data.hero.loc.x != core.getHeroLoc('x') || data.hero.loc.y != core.getHeroLoc('y'))
-        return alert("楼层或坐标不一致!");
-
-    core.status.tempRoute = route;
+  if (!data) {
+    core.playSound("操作失败");
+    return core.drawTip("无效的存档");
+  }
+  var route = core.decodeRoute(data.route);
+  if (core.status.tempRoute) {
+    var remainRoute = core.subarray(route, core.status.tempRoute);
+    if (remainRoute == null)
+      return alert(
+        "无法接续播放录像!\n该存档必须是前一个选择的存档的后续内容。"
+      );
+    delete core.status.tempRoute;
     core.ui.closePanel();
-    core.drawText("\t[步骤2]请选择第二个存档。\n\r[yellow]该存档必须是前一个存档的后续。\r\n将尝试播放到此存档。", function () {
-        core.status.event.id = 'replayRemain';
-        core.lockControl();
-        var saveIndex = core.saves.saveIndex;
-        var page = parseInt((saveIndex - 1) / 5), offset = saveIndex - 5 * page;
-        core.ui._drawSLPanel(10 * page + offset);
-    });
-}
+    core.startReplay(remainRoute);
+    core.drawTip("接续播放录像");
+    return;
+  } else if (
+    data.floorId != core.status.floorId ||
+    data.hero.loc.x != core.getHeroLoc("x") ||
+    data.hero.loc.y != core.getHeroLoc("y")
+  )
+    return alert("楼层或坐标不一致!");
+
+  core.status.tempRoute = route;
+  core.ui.closePanel();
+  core.drawText(
+    "\t[步骤2]请选择第二个存档。\n\r[yellow]该存档必须是前一个存档的后续。\r\n将尝试播放到此存档。",
+    function () {
+      core.status.event.id = "replayRemain";
+      core.lockControl();
+      var saveIndex = core.saves.saveIndex;
+      var page = parseInt((saveIndex - 1) / 5),
+        offset = saveIndex - 5 * page;
+      core.ui._drawSLPanel(10 * page + offset);
+    }
+  );
+};
 
 control.prototype._doSL_replaySince_afterGet = function (id, data) {
-    if (data.floorId != core.status.floorId || data.hero.loc.x != core.getHeroLoc('x') || data.hero.loc.y != core.getHeroLoc('y'))
-        return alert("楼层或坐标不一致!");
-    if (!data.__toReplay__) return alert('该存档没有剩余录像!');
-    core.ui.closePanel();
-    core.startReplay(core.decodeRoute(data.__toReplay__));
-    core.drawTip("播放存档剩余录像");
-    return;
-}
+  if (
+    data.floorId != core.status.floorId ||
+    data.hero.loc.x != core.getHeroLoc("x") ||
+    data.hero.loc.y != core.getHeroLoc("y")
+  )
+    return alert("楼层或坐标不一致!");
+  if (!data.__toReplay__) return alert("该存档没有剩余录像!");
+  core.ui.closePanel();
+  core.startReplay(core.decodeRoute(data.__toReplay__));
+  core.drawTip("播放存档剩余录像");
+  return;
+};
 
 ////// 同步存档到服务器 //////
 control.prototype.syncSave = function (type) {
-    core.ui.drawWaiting("正在同步,请稍候...");
-    var callback = function (saves) {
-        core.control._syncSave_http(type, saves);
-    }
-    if (type == 'all') core.getAllSaves(callback);
-    else core.getSave(core.saves.saveIndex, callback);
-}
+  core.ui.drawWaiting("正在同步,请稍候...");
+  var callback = function (saves) {
+    core.control._syncSave_http(type, saves);
+  };
+  if (type == "all") core.getAllSaves(callback);
+  else core.getSave(core.saves.saveIndex, callback);
+};
 
 control.prototype._syncSave_http = function (type, saves) {
-    if (!saves) return core.drawText("没有要同步的存档");
-    var formData = new FormData();
-    formData.append('type', 'save');
-    formData.append('name', core.firstData.name);
-    formData.append('data', LZString.compressToBase64(JSON.stringify(saves)));
-    formData.append('shorten', '1');
+  if (!saves) return core.drawText("没有要同步的存档");
+  var formData = new FormData();
+  formData.append("type", "save");
+  formData.append("name", core.firstData.name);
+  formData.append("data", LZString.compressToBase64(JSON.stringify(saves)));
+  formData.append("shorten", "1");
 
-    core.http("POST", "/games/sync.php", formData, function (data) {
-        var response = JSON.parse(data);
-        if (response.code < 0) {
-            core.drawText("出错啦!\n无法同步存档到服务器。\n错误原因:" + response.msg);
-        }
-        else {
-            core.drawText((type == 'all' ? "所有存档" : "存档" + core.saves.saveIndex) + "同步成功!\n\n您的存档编号+密码: \r[yellow]"
-                + response.code + response.msg
-                + "\r\n\n请牢记以上信息(如截图等),在从服务器\n同步存档时使用。\n\r[yellow]另外请注意,存档同步只会保存一个月的时间。\r")
-        }
-    }, function (e) {
-        core.drawText("出错啦!\n无法同步存档到服务器。\n错误原因:" + e);
-    })
-}
+  core.http(
+    "POST",
+    "/games/sync.php",
+    formData,
+    function (data) {
+      var response = JSON.parse(data);
+      if (response.code < 0) {
+        core.drawText(
+          "出错啦!\n无法同步存档到服务器。\n错误原因:" + response.msg
+        );
+      } else {
+        core.drawText(
+          (type == "all" ? "所有存档" : "存档" + core.saves.saveIndex) +
+            "同步成功!\n\n您的存档编号+密码: \r[yellow]" +
+            response.code +
+            response.msg +
+            "\r\n\n请牢记以上信息(如截图等),在从服务器\n同步存档时使用。\n\r[yellow]另外请注意,存档同步只会保存一个月的时间。\r"
+        );
+      }
+    },
+    function (e) {
+      core.drawText("出错啦!\n无法同步存档到服务器。\n错误原因:" + e);
+    }
+  );
+};
 
 ////// 从服务器加载存档 //////
 control.prototype.syncLoad = function () {
-    core.myprompt("请输入存档编号+密码", null, function (idpassword) {
-        if (!idpassword) return core.ui._drawSyncSave();
-        if (!/^\d{6}\w{4}$/.test(idpassword) && !/^\d{4}\w{3}$/.test(idpassword)) {
-            core.drawText("不合法的存档编号+密码!");
-            return;
-        }
-        core.ui.drawWaiting("正在同步,请稍候...");
-        if (idpassword.length == 7) {
-            core.control._syncLoad_http(idpassword.substring(0, 4), idpassword.substring(4));
-        } else {
-            core.control._syncLoad_http(idpassword.substring(0, 6), idpassword.substring(6));
-        }
-    });
-}
+  core.myprompt("请输入存档编号+密码", null, function (idpassword) {
+    if (!idpassword) return core.ui._drawSyncSave();
+    if (!/^\d{6}\w{4}$/.test(idpassword) && !/^\d{4}\w{3}$/.test(idpassword)) {
+      core.drawText("不合法的存档编号+密码!");
+      return;
+    }
+    core.ui.drawWaiting("正在同步,请稍候...");
+    if (idpassword.length == 7) {
+      core.control._syncLoad_http(
+        idpassword.substring(0, 4),
+        idpassword.substring(4)
+      );
+    } else {
+      core.control._syncLoad_http(
+        idpassword.substring(0, 6),
+        idpassword.substring(6)
+      );
+    }
+  });
+};
 
 control.prototype._syncLoad_http = function (id, password) {
-    var formData = new FormData();
-    formData.append('type', 'load');
-    formData.append('name', core.firstData.name);
-    formData.append('id', id);
-    formData.append('password', password);
+  var formData = new FormData();
+  formData.append("type", "load");
+  formData.append("name", core.firstData.name);
+  formData.append("id", id);
+  formData.append("password", password);
 
-    core.http("POST", "/games/sync.php", formData, function (data) {
-        var response = JSON.parse(data);
-        if (response.code == 0) {
-            var msg = null;
-            try {
-                msg = JSON.parse(LZString.decompressFromBase64(response.msg));
-            } catch (e) { }
-            if (!msg) {
-                try {
-                    msg = JSON.parse(response.msg);
-                } catch (e) { }
-            }
-            if (msg) {
-                core.control._syncLoad_write(msg);
-            } else {
-                core.drawText("出错啦!\n存档解析失败!");
-            }
+  core.http(
+    "POST",
+    "/games/sync.php",
+    formData,
+    function (data) {
+      var response = JSON.parse(data);
+      if (response.code == 0) {
+        var msg = null;
+        try {
+          msg = JSON.parse(LZString.decompressFromBase64(response.msg));
+        } catch (e) {}
+        if (!msg) {
+          try {
+            msg = JSON.parse(response.msg);
+          } catch (e) {}
         }
-        else {
-            core.drawText("出错啦!\n无法从服务器同步存档。\n错误原因:" + response.msg);
+        if (msg) {
+          core.control._syncLoad_write(msg);
+        } else {
+          core.drawText("出错啦!\n存档解析失败!");
         }
-    }, function (e) {
-        core.drawText("出错啦!\n无法从服务器同步存档。\n错误原因:" + e);
-    });
-}
+      } else {
+        core.drawText(
+          "出错啦!\n无法从服务器同步存档。\n错误原因:" + response.msg
+        );
+      }
+    },
+    function (e) {
+      core.drawText("出错啦!\n无法从服务器同步存档。\n错误原因:" + e);
+    }
+  );
+};
 
 control.prototype._syncLoad_write = function (data) {
-    if (data instanceof Array) {
-        core.status.event.selection = 1;
-        core.ui.drawConfirmBox("所有本地存档都将被覆盖,确认?", function () {
-            for (var i = 1; i <= 5 * (main.savePages || 30); i++) {
-                if (i <= data.length)
-                    core.setLocalForage("save" + i, data[i - 1]);
-                else if (core.saves.ids[i])
-                    core.removeLocalForage("save" + i);
-            }
-            core.ui.closePanel();
-            core.drawText("同步成功!\n你的本地所有存档均已被覆盖。");
-        }, function () {
-            core.status.event.selection = 0;
-            core.ui._drawSyncSave();
-        });
-    }
-    else {
-        // 只覆盖单存档
-        core.setLocalForage("save" + core.saves.saveIndex, data, function () {
-            core.drawText("同步成功!\n单存档已覆盖至存档" + core.saves.saveIndex);
-        });
-    }
-}
+  if (data instanceof Array) {
+    core.status.event.selection = 1;
+    core.ui.drawConfirmBox(
+      "所有本地存档都将被覆盖,确认?",
+      function () {
+        for (var i = 1; i <= 5 * (main.savePages || 30); i++) {
+          if (i <= data.length) core.setLocalForage("save" + i, data[i - 1]);
+          else if (core.saves.ids[i]) core.removeLocalForage("save" + i);
+        }
+        core.ui.closePanel();
+        core.drawText("同步成功!\n你的本地所有存档均已被覆盖。");
+      },
+      function () {
+        core.status.event.selection = 0;
+        core.ui._drawSyncSave();
+      }
+    );
+  } else {
+    // 只覆盖单存档
+    core.setLocalForage("save" + core.saves.saveIndex, data, function () {
+      core.drawText("同步成功!\n单存档已覆盖至存档" + core.saves.saveIndex);
+    });
+  }
+};
 
 ////// 存档到本地 //////
 control.prototype.saveData = function () {
-    return this.controldata.saveData();
-}
+  return this.controldata.saveData();
+};
 
 ////// 从本地读档 //////
 control.prototype.loadData = function (data, callback) {
-    return this.controldata.loadData(data, callback);
-}
+  return this.controldata.loadData(data, callback);
+};
 
 control.prototype.getSave = function (index, callback) {
-    if (index == 0) {
-        // --- 自动存档先从缓存中获取
-        if (core.saves.autosave.data != null)
-            callback(core.saves.autosave.data);
-        else {
-            core.getLocalForage("autoSave", null, function (data) {
-                if (data != null) {
-                    core.saves.autosave.data = data;
-                    if (!(core.saves.autosave.data instanceof Array)) {
-                        core.saves.autosave.data = [core.saves.autosave.data];
-                    }
-                    core.saves.autosave.now = core.saves.autosave.data.length;
-                }
-                callback(core.saves.autosave.data);
-            }, function (err) {
-                console.error(err);
-                callback(null);
-            });
+  if (index == 0) {
+    // --- 自动存档先从缓存中获取
+    if (core.saves.autosave.data != null) callback(core.saves.autosave.data);
+    else {
+      core.getLocalForage(
+        "autoSave",
+        null,
+        function (data) {
+          if (data != null) {
+            core.saves.autosave.data = data;
+            if (!(core.saves.autosave.data instanceof Array)) {
+              core.saves.autosave.data = [core.saves.autosave.data];
+            }
+            core.saves.autosave.now = core.saves.autosave.data.length;
+          }
+          callback(core.saves.autosave.data);
+        },
+        function (err) {
+          console.error(err);
+          callback(null);
         }
-        return;
+      );
     }
-    core.getLocalForage("save" + index, null, function (data) {
-        if (callback) callback(data);
-    }, function (err) {
-        console.error(err);
-        if (callback) callback(null);
-    });
-}
+    return;
+  }
+  core.getLocalForage(
+    "save" + index,
+    null,
+    function (data) {
+      if (callback) callback(data);
+    },
+    function (err) {
+      console.error(err);
+      if (callback) callback(null);
+    }
+  );
+};
 
 control.prototype.getSaves = function (ids, callback) {
-    if (!(ids instanceof Array)) return this.getSave(ids, callback);
-    var count = ids.length, data = {};
-    for (var i = 0; i < ids.length; ++i) {
-        (function (i) {
-            core.getSave(ids[i], function (result) {
-                data[i] = result;
-                if (Object.keys(data).length == count)
-                    callback(data);
-            })
-        })(i);
-    }
-}
+  if (!(ids instanceof Array)) return this.getSave(ids, callback);
+  var count = ids.length,
+    data = {};
+  for (var i = 0; i < ids.length; ++i) {
+    (function (i) {
+      core.getSave(ids[i], function (result) {
+        data[i] = result;
+        if (Object.keys(data).length == count) callback(data);
+      });
+    })(i);
+  }
+};
 
 control.prototype.getAllSaves = function (callback) {
-    var ids = Object.keys(core.saves.ids).filter(function (x) { return x != 0; })
-        .sort(function (a, b) { return a - b; }), saves = [];
-    this.getSaves(ids, function (data) {
-        for (var i = 0; i < ids.length; ++i) {
-            if (data[i] != null)
-                saves.push(data[i]);
-        }
-        callback(saves);
-    });
-}
+  var ids = Object.keys(core.saves.ids)
+      .filter(function (x) {
+        return x != 0;
+      })
+      .sort(function (a, b) {
+        return a - b;
+      }),
+    saves = [];
+  this.getSaves(ids, function (data) {
+    for (var i = 0; i < ids.length; ++i) {
+      if (data[i] != null) saves.push(data[i]);
+    }
+    callback(saves);
+  });
+};
 
 ////// 获得所有存在存档的存档位 //////
 control.prototype.getSaveIndexes = function (callback) {
-    var indexes = {};
-    core.keysLocalForage(function (err, keys) {
-        if (err) {
-            console.error(err);
-            return callback(indexes);
-        }
-        keys.forEach(function (key) {
-            core.control._getSaveIndexes_getIndex(indexes, key);
-        });
-        callback(indexes);
+  var indexes = {};
+  core.keysLocalForage(function (err, keys) {
+    if (err) {
+      console.error(err);
+      return callback(indexes);
+    }
+    keys.forEach(function (key) {
+      core.control._getSaveIndexes_getIndex(indexes, key);
     });
-}
+    callback(indexes);
+  });
+};
 
 control.prototype._getSaveIndexes_getIndex = function (indexes, name) {
-    var e = new RegExp('^' + core.firstData.name + "_(save\\d+|autoSave)$").exec(name);
-    if (e) {
-        if (e[1] == 'autoSave') indexes[0] = true;
-        else indexes[parseInt(e[1].substring(4))] = true;
-    }
-}
+  var e = new RegExp("^" + core.firstData.name + "_(save\\d+|autoSave)$").exec(
+    name
+  );
+  if (e) {
+    if (e[1] == "autoSave") indexes[0] = true;
+    else indexes[parseInt(e[1].substring(4))] = true;
+  }
+};
 
 ////// 判断某个存档位是否存在存档 //////
 control.prototype.hasSave = function (index) {
-    return core.saves.ids[index] || false;
-}
+  return core.saves.ids[index] || false;
+};
 
 ////// 删除某个存档
 control.prototype.removeSave = function (index, callback) {
-    if (index == 0 || index == "autoSave") {
-        index = "autoSave";
-        core.removeLocalForage(index, function () {
-            core.saves.autosave.data = null;
-            core.saves.autosave.updated = false;
-            if (callback) callback();
-        });
-        return;
-    }
-    core.removeLocalForage("save" + index, function () {
-        core.saves.favorite = core.saves.favorite.filter(function (i) { return core.hasSave(i); });
-        delete core.saves.favoriteName[index];
-        core.control._updateFavoriteSaves();
-        if (callback) callback();
-    }, function () {
-        core.playSound('操作失败');
-        core.drawTip("无法删除存档!");
-        if (callback) callback();
+  if (index == 0 || index == "autoSave") {
+    index = "autoSave";
+    core.removeLocalForage(index, function () {
+      core.saves.autosave.data = null;
+      core.saves.autosave.updated = false;
+      if (callback) callback();
     });
-}
+    return;
+  }
+  core.removeLocalForage(
+    "save" + index,
+    function () {
+      core.saves.favorite = core.saves.favorite.filter(function (i) {
+        return core.hasSave(i);
+      });
+      delete core.saves.favoriteName[index];
+      core.control._updateFavoriteSaves();
+      if (callback) callback();
+    },
+    function () {
+      core.playSound("操作失败");
+      core.drawTip("无法删除存档!");
+      if (callback) callback();
+    }
+  );
+};
 
 ////// 读取收藏信息
 control.prototype._loadFavoriteSaves = function () {
-    core.saves.favorite = core.getLocalStorage("favorite", []);
-    // --- 移除不存在的收藏
-    core.saves.favorite = core.saves.favorite.filter(function (i) { return core.hasSave(i); });
-    core.saves.favoriteName = core.getLocalStorage("favoriteName", {});
-}
+  core.saves.favorite = core.getLocalStorage("favorite", []);
+  // --- 移除不存在的收藏
+  core.saves.favorite = core.saves.favorite.filter(function (i) {
+    return core.hasSave(i);
+  });
+  core.saves.favoriteName = core.getLocalStorage("favoriteName", {});
+};
 
 control.prototype._updateFavoriteSaves = function () {
-    core.setLocalStorage("favorite", core.saves.favorite);
-    core.setLocalStorage("favoriteName", core.saves.favoriteName);
-}
+  core.setLocalStorage("favorite", core.saves.favorite);
+  core.setLocalStorage("favoriteName", core.saves.favoriteName);
+};
 
 // ------ 属性,状态,位置,buff,变量,锁定控制等 ------ //
 
 ////// 设置勇士属性 //////
 control.prototype.setStatus = function (name, value) {
-    if (!core.status.hero) return;
-    if (name == 'x' || name == 'y' || name == 'direction')
-        this.setHeroLoc(name, value);
-    else
-        core.status.hero[name] = value;
-}
+  if (!core.status.hero) return;
+  if (name == "x" || name == "y" || name == "direction")
+    this.setHeroLoc(name, value);
+  else core.status.hero[name] = value;
+};
 
 ////// 增减勇士属性 //////
 control.prototype.addStatus = function (name, value) {
-    this.setStatus(name, this.getStatus(name) + value);
-}
+  this.setStatus(name, this.getStatus(name) + value);
+};
 
 ////// 获得勇士属性 //////
 control.prototype.getStatus = function (name) {
-    if (!core.status.hero) return null;
-    if (name == 'x' || name == 'y' || name == 'direction')
-        return this.getHeroLoc(name);
-    if (main.mode == 'editor' && !core.hasFlag('__statistics__')) {
-        return data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.hero[name];
-    }
-    return core.status.hero[name];
-}
+  if (!core.status.hero) return null;
+  if (name == "x" || name == "y" || name == "direction")
+    return this.getHeroLoc(name);
+  if (main.mode == "editor" && !core.hasFlag("__statistics__")) {
+    return data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.hero[name];
+  }
+  return core.status.hero[name];
+};
 
 ////// 从status中获得属性,如果不存在则从勇士属性中获取 //////
 control.prototype.getStatusOrDefault = function (status, name) {
-    if (status && name in status)
-        return Math.floor(status[name]);
-    return Math.floor(this.getStatus(name));
-}
+  if (status && name in status) return Math.floor(status[name]);
+  return Math.floor(this.getStatus(name));
+};
 
 ////// 获得勇士实际属性(增幅后的) //////
 control.prototype.getRealStatus = function (name) {
-    return this.getRealStatusOrDefault(null, name);
-}
+  return this.getRealStatusOrDefault(null, name);
+};
 
 ////// 从status中获得实际属性(增幅后的),如果不存在则从勇士属性中获取 //////
 control.prototype.getRealStatusOrDefault = function (status, name) {
-    return Math.floor(this.getStatusOrDefault(status, name) * this.getBuff(name));
-}
+  return Math.floor(this.getStatusOrDefault(status, name) * this.getBuff(name));
+};
 
 ////// 获得勇士原始属性(无装备和衰弱影响) //////
 control.prototype.getNakedStatus = function (name) {
-    var value = this.getStatus(name);
-    if (value == null) return value;
-    // 装备增幅
-    core.status.hero.equipment.forEach(function (v) {
-        if (!v || !(core.material.items[v] || {}).equip) return;
-        value -= core.material.items[v].equip.value[name] || 0;
-    });
-    // 衰弱扣除
-    if (core.hasFlag('weak') && core.values.weakValue >= 1 && (name == 'atk' || name == 'def')) {
-        value += core.values.weakValue;
-    }
-    return value;
-}
+  var value = this.getStatus(name);
+  if (value == null) return value;
+  // 装备增幅
+  core.status.hero.equipment.forEach(function (v) {
+    if (!v || !(core.material.items[v] || {}).equip) return;
+    value -= core.material.items[v].equip.value[name] || 0;
+  });
+  // 衰弱扣除
+  if (
+    core.hasFlag("weak") &&
+    core.values.weakValue >= 1 &&
+    (name == "atk" || name == "def")
+  ) {
+    value += core.values.weakValue;
+  }
+  return value;
+};
 
 ////// 获得某个属性的名字 //////
 control.prototype.getStatusLabel = function (name) {
-    if (this.controldata.getStatusLabel) {
-        return this.controldata.getStatusLabel(name) || name;
-    }
-    return {
-        name: "名称", lv: "等级", hpmax: "生命上限", hp: "生命", manamax: "魔力上限", mana: "魔力",
-        atk: "攻击", def: "防御", mdef: "护盾", money: "金币", exp: "经验", point: "加点", steps: "步数"
-    }[name] || name;
-}
+  if (this.controldata.getStatusLabel) {
+    return this.controldata.getStatusLabel(name) || name;
+  }
+  return (
+    {
+      name: "名称",
+      lv: "等级",
+      hpmax: "生命上限",
+      hp: "生命",
+      manamax: "魔力上限",
+      mana: "魔力",
+      atk: "攻击",
+      def: "防御",
+      mdef: "护盾",
+      money: "金币",
+      exp: "经验",
+      point: "加点",
+      steps: "步数",
+    }[name] || name
+  );
+};
 
 ////// 设置某个属性的增幅值 //////
 control.prototype.setBuff = function (name, value) {
-    // 仅保留三位有效buff值
-    value = parseFloat(value.toFixed(3));
-    this.setFlag('__' + name + '_buff__', value);
-}
+  // 仅保留三位有效buff值
+  value = parseFloat(value.toFixed(3));
+  this.setFlag("__" + name + "_buff__", value);
+};
 
 ////// 加减某个属性的增幅值 //////
 control.prototype.addBuff = function (name, value) {
-    var buff = this.getBuff(name) + value;
-    // 仅保留三位有效buff值
-    buff = parseFloat(buff.toFixed(3));
-    this.setFlag('__' + name + '_buff__', buff);
-}
+  var buff = this.getBuff(name) + value;
+  // 仅保留三位有效buff值
+  buff = parseFloat(buff.toFixed(3));
+  this.setFlag("__" + name + "_buff__", buff);
+};
 
 ////// 获得某个属性的增幅值 //////
 control.prototype.getBuff = function (name) {
-    return core.getFlag('__' + name + '_buff__', 1);
-}
+  return core.getFlag("__" + name + "_buff__", 1);
+};
 
 ////// 获得或移除毒衰咒效果 //////
 control.prototype.triggerDebuff = function (action, type) {
-    return this.controldata.triggerDebuff(action, type);
-}
+  return this.controldata.triggerDebuff(action, type);
+};
 
 ////// 设置勇士的位置 //////
 control.prototype.setHeroLoc = function (name, value, noGather) {
-    if (!core.status.hero) return;
-    core.status.hero.loc[name] = value;
-    if ((name == 'x' || name == 'y') && !noGather) {
-        this.gatherFollowers();
-    }
-    core.ui.drawStatusBar();
-}
+  if (!core.status.hero) return;
+  core.status.hero.loc[name] = value;
+  if ((name == "x" || name == "y") && !noGather) {
+    this.gatherFollowers();
+  }
+  core.ui.drawStatusBar();
+};
 
 ////// 获得勇士的位置 //////
 control.prototype.getHeroLoc = function (name) {
-    if (!core.status.hero) return;
-    if (main.mode == 'editor') {
-        if (name == null) return data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.hero.loc;
-        return data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.hero.loc[name];
-    }
-    if (name == null) return core.status.hero.loc;
-    return core.status.hero.loc[name];
-}
+  if (!core.status.hero) return;
+  if (main.mode == "editor") {
+    if (name == null)
+      return data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.hero.loc;
+    return data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.hero.loc[name];
+  }
+  if (name == null) return core.status.hero.loc;
+  return core.status.hero.loc[name];
+};
 
 ////// 获得某个等级的名称 //////
 control.prototype.getLvName = function (lv) {
-    if (!core.status.hero) return null;
-    if (lv == null) lv = core.status.hero.lv;
-    return ((core.firstData.levelUp || [])[lv - 1] || {}).title || lv;
-}
+  if (!core.status.hero) return null;
+  if (lv == null) lv = core.status.hero.lv;
+  return ((core.firstData.levelUp || [])[lv - 1] || {}).title || lv;
+};
 
 ////// 获得下个等级所需经验;如果不存在下个等级,返回null。 //////
 control.prototype.getNextLvUpNeed = function () {
-    if (!core.status.hero) return null;
-    if (core.status.hero.lv >= core.firstData.levelUp.length) return null;
-    var need = core.calValue(core.firstData.levelUp[core.status.hero.lv].need);
-    if (core.flags.statusBarItems.indexOf('levelUpLeftMode') >= 0)
-        return Math.max(need - core.getStatus('exp'), 0);
-    else return need;
-}
+  if (!core.status.hero) return null;
+  if (core.status.hero.lv >= core.firstData.levelUp.length) return null;
+  var need = core.calValue(core.firstData.levelUp[core.status.hero.lv].need);
+  if (core.flags.statusBarItems.indexOf("levelUpLeftMode") >= 0)
+    return Math.max(need - core.getStatus("exp"), 0);
+  else return need;
+};
 
 ////// 设置某个自定义变量或flag //////
 control.prototype.setFlag = function (name, value) {
-    if (value == null) return this.removeFlag(name);
-    if (!core.status.hero) return;
-    core.status.hero.flags[name] = value;
-}
+  if (value == null) return this.removeFlag(name);
+  if (!core.status.hero) return;
+  core.status.hero.flags[name] = value;
+};
 
 ////// 增加某个flag数值 //////
 control.prototype.addFlag = function (name, value) {
-    if (!core.status.hero) return;
-    core.setFlag(name, core.getFlag(name, 0) + value);
-}
+  if (!core.status.hero) return;
+  core.setFlag(name, core.getFlag(name, 0) + value);
+};
 
 ////// 获得某个自定义变量或flag //////
 control.prototype.getFlag = function (name, defaultValue) {
-    if (!core.status.hero) return defaultValue;
-    var value = core.status.hero.flags[name];
-    return value != null ? value : defaultValue;
-}
+  if (!core.status.hero) return defaultValue;
+  var value = core.status.hero.flags[name];
+  return value != null ? value : defaultValue;
+};
 
 ////// 是否存在某个自定义变量或flag,且值为true //////
 control.prototype.hasFlag = function (name) {
-    return !!core.getFlag(name);
-}
+  return !!core.getFlag(name);
+};
 
 ////// 删除某个自定义变量或flag //////
 control.prototype.removeFlag = function (name) {
-    if (!core.status.hero) return;
-    delete core.status.hero.flags[name];
-}
+  if (!core.status.hero) return;
+  delete core.status.hero.flags[name];
+};
 
 ////// 获得某个点的独立开关 //////
 control.prototype.getSwitch = function (x, y, floorId, name, defaultValue) {
-    var prefix = [floorId || core.status.floorId || ":f", x != null ? x : "x", y != null ? y : "y"].join("@");
-    return this.getFlag(prefix + "@" + name, defaultValue);
-}
+  var prefix = [
+    floorId || core.status.floorId || ":f",
+    x != null ? x : "x",
+    y != null ? y : "y",
+  ].join("@");
+  return this.getFlag(prefix + "@" + name, defaultValue);
+};
 
 ////// 设置某个点的独立开关 //////
 control.prototype.setSwitch = function (x, y, floorId, name, value) {
-    var prefix = [floorId || core.status.floorId || ":f", x != null ? x : "x", y != null ? y : "y"].join("@");
-    return this.setFlag(prefix + "@" + name, value);
-}
+  var prefix = [
+    floorId || core.status.floorId || ":f",
+    x != null ? x : "x",
+    y != null ? y : "y",
+  ].join("@");
+  return this.setFlag(prefix + "@" + name, value);
+};
 
 ////// 增加某个点的独立开关 //////
 control.prototype.addSwitch = function (x, y, floorId, name, value) {
-    var prefix = [floorId || core.status.floorId || ":f", x != null ? x : "x", y != null ? y : "y"].join("@");
-    return this.addFlag(prefix + "@" + name, value);
-}
+  var prefix = [
+    floorId || core.status.floorId || ":f",
+    x != null ? x : "x",
+    y != null ? y : "y",
+  ].join("@");
+  return this.addFlag(prefix + "@" + name, value);
+};
 
 ////// 判定某个点的独立开关 //////
 control.prototype.hasSwitch = function (x, y, floorId, name) {
-    var prefix = [floorId || core.status.floorId || ":f", x != null ? x : "x", y != null ? y : "y"].join("@");
-    return this.hasFlag(prefix + "@" + name);
-}
+  var prefix = [
+    floorId || core.status.floorId || ":f",
+    x != null ? x : "x",
+    y != null ? y : "y",
+  ].join("@");
+  return this.hasFlag(prefix + "@" + name);
+};
 
 ////// 删除某个点的独立开关 //////
 control.prototype.removeSwitch = function (x, y, floorId, name) {
-    var prefix = [floorId || core.status.floorId || ":f", x != null ? x : "x", y != null ? y : "y"].join("@");
-    return this.removeFlag(prefix + "@" + name);
-}
+  var prefix = [
+    floorId || core.status.floorId || ":f",
+    x != null ? x : "x",
+    y != null ? y : "y",
+  ].join("@");
+  return this.removeFlag(prefix + "@" + name);
+};
 
 ////// 锁定状态栏,常常用于事件处理 //////
 control.prototype.lockControl = function () {
-    core.status.lockControl = true;
-}
+  core.status.lockControl = true;
+};
 
 ////// 解锁状态栏 //////
 control.prototype.unlockControl = function () {
-    core.status.lockControl = false;
-}
+  core.status.lockControl = false;
+};
 
 ////// 开启debug模式 //////
 control.prototype.debug = function () {
-    core.setFlag('debug', true);
-    core.drawText("\t[调试模式开启]此模式下按住Ctrl键(或Ctrl+Shift键)可以穿墙并忽略一切事件。\n此模式下将无法上传成绩。");
-}
+  core.setFlag("debug", true);
+  core.drawText(
+    "\t[调试模式开启]此模式下按住Ctrl键(或Ctrl+Shift键)可以穿墙并忽略一切事件。\n此模式下将无法上传成绩。"
+  );
+};
 
 control.prototype._bindRoutePush = function () {
-    core.status.route.push = function (element) {
-        // 忽视移动、转向、瞬移
-        if (["up", "down", "left", "right", "turn"].indexOf(element) < 0 && !element.startsWith("move:")) {
-            core.clearRouteFolding();
-        }
-        Array.prototype.push.call(core.status.route, element);
+  core.status.route.push = function (element) {
+    // 忽视移动、转向、瞬移
+    if (
+      ["up", "down", "left", "right", "turn"].indexOf(element) < 0 &&
+      !element.startsWith("move:")
+    ) {
+      core.clearRouteFolding();
     }
-}
+    Array.prototype.push.call(core.status.route, element);
+  };
+};
 
 ////// 清除录像折叠信息 //////
 control.prototype.clearRouteFolding = function () {
-    core.status.routeFolding = {};
-}
+  core.status.routeFolding = {};
+};
 
 ////// 检查录像折叠 //////
 control.prototype.checkRouteFolding = function () {
-    // 未开启、未开始游戏、录像播放中、正在事件中:不执行
-    if (!core.flags.enableRouteFolding || !core.isPlaying() || core.isReplaying() || core.status.event.id) {
-        return this.clearRouteFolding();
+  // 未开启、未开始游戏、录像播放中、正在事件中:不执行
+  if (
+    !core.flags.enableRouteFolding ||
+    !core.isPlaying() ||
+    core.isReplaying() ||
+    core.status.event.id
+  ) {
+    return this.clearRouteFolding();
+  }
+  var hero = core.clone(core.status.hero, function (name, value) {
+    return name != "steps" && typeof value == "number";
+  });
+  var index = [
+    core.getHeroLoc("x"),
+    core.getHeroLoc("y"),
+    core.getHeroLoc("direction").charAt(0),
+  ].join(",");
+  core.status.routeFolding = core.status.routeFolding || {};
+  if (core.status.routeFolding[index]) {
+    var one = core.status.routeFolding[index];
+    if (core.same(one.hero, hero) && one.length < core.status.route.length) {
+      Object.keys(core.status.routeFolding).forEach(function (v) {
+        if (core.status.routeFolding[v].length >= one.length)
+          delete core.status.routeFolding[v];
+      });
+      core.status.route = core.status.route.slice(0, one.length);
+      this._bindRoutePush();
     }
-    var hero = core.clone(core.status.hero, function (name, value) {
-        return name != 'steps' && typeof value == 'number';
-    });
-    var index = [core.getHeroLoc('x'), core.getHeroLoc('y'), core.getHeroLoc('direction').charAt(0)].join(',');
-    core.status.routeFolding = core.status.routeFolding || {};
-    if (core.status.routeFolding[index]) {
-        var one = core.status.routeFolding[index];
-        if (core.same(one.hero, hero) && one.length < core.status.route.length) {
-            Object.keys(core.status.routeFolding).forEach(function (v) {
-                if (core.status.routeFolding[v].length >= one.length) delete core.status.routeFolding[v];
-            });
-            core.status.route = core.status.route.slice(0, one.length);
-            this._bindRoutePush();
-        }
-    }
-    core.status.routeFolding[index] = { hero: hero, length: core.status.route.length };
-}
+  }
+  core.status.routeFolding[index] = {
+    hero: hero,
+    length: core.status.route.length,
+  };
+};
 
 // ------ 天气,色调,BGM ------ //
 
 control.prototype.getMappedName = function (name) {
-    return core.getFlag('__nameMap__', {})[name] || (main.nameMap || {})[name] || name;
-}
+  return (
+    core.getFlag("__nameMap__", {})[name] || (main.nameMap || {})[name] || name
+  );
+};
 
 ////// 更改天气效果 //////
 control.prototype.setWeather = function (type, level) {
-    // 非雨雪
-    if (type == null || !this.weathers[type]) {
-        core.deleteCanvas('weather')
-        core.animateFrame.weather.type = null;
-        core.animateFrame.weather.nodes = [];
-        return;
-    }
-    if (level == null) level = core.animateFrame.weather.level;
-    level = core.clamp(parseInt(level) || 5, 1, 10);
-    // 当前天气:则忽略
-    if (type == core.animateFrame.weather.type && level == core.animateFrame.weather.level) return;
-
-    // 计算当前的宽高
-    core.createCanvas('weather', 0, 0, core._PX_, core._PY_, 80);
-    core.setOpacity('weather', 1.0);
-    core.animateFrame.weather.type = type;
-    core.animateFrame.weather.level = level;
+  // 非雨雪
+  if (type == null || !this.weathers[type]) {
+    core.deleteCanvas("weather");
+    core.animateFrame.weather.type = null;
     core.animateFrame.weather.nodes = [];
-    try {
-        core.doFunc(this.weathers[type].initFunc, this, level);
-    } catch (e) {
-        console.error(e);
-        console.error("ERROR in weather[" + type + "]:已自动注销该项。");
-        core.unregisterWeather(type);
-    }
-}
+    return;
+  }
+  if (level == null) level = core.animateFrame.weather.level;
+  level = core.clamp(parseInt(level) || 5, 1, 10);
+  // 当前天气:则忽略
+  if (
+    type == core.animateFrame.weather.type &&
+    level == core.animateFrame.weather.level
+  )
+    return;
+
+  // 计算当前的宽高
+  core.createCanvas("weather", 0, 0, core._PX_, core._PY_, 80);
+  core.setOpacity("weather", 1.0);
+  core.animateFrame.weather.type = type;
+  core.animateFrame.weather.level = level;
+  core.animateFrame.weather.nodes = [];
+  try {
+    core.doFunc(this.weathers[type].initFunc, this, level);
+  } catch (e) {
+    console.error(e);
+    console.error("ERROR in weather[" + type + "]:已自动注销该项。");
+    core.unregisterWeather(type);
+  }
+};
 
 ////// 注册一个天气 //////
 // name为天气类型,如 sun, rain, snow 等
 // initFunc 为设置为此天气时的初始化,接受level参数
 // frameFunc 为该天气下每帧的效果,接受和timestamp参数(从页面加载完毕到当前经过的时间)
 control.prototype.registerWeather = function (name, initFunc, frameFunc) {
-    this.unregisterWeather(name);
-    this.weathers[name] = { initFunc: initFunc, frameFunc: frameFunc };
-}
+  this.unregisterWeather(name);
+  this.weathers[name] = { initFunc: initFunc, frameFunc: frameFunc };
+};
 
 ////// 取消注册一个天气 //////
 control.prototype.unregisterWeather = function (name) {
-    delete this.weathers[name];
-    if (core.animateFrame.weather.type == name) {
-        this.setWeather(null);
-    }
-}
+  delete this.weathers[name];
+  if (core.animateFrame.weather.type == name) {
+    this.setWeather(null);
+  }
+};
 
 control.prototype._weather_rain = function (level) {
-    var number = level * parseInt(20 * core.bigmap.width * core.bigmap.height / (core._WIDTH_ * core._HEIGHT_));
-    for (var a = 0; a < number; a++) {
-        core.animateFrame.weather.nodes.push({
-            'x': Math.random() * core.bigmap.width * 32,
-            'y': Math.random() * core.bigmap.height * 32,
-            'l': Math.random() * 2.5,
-            'xs': -4 + Math.random() * 4 + 2,
-            'ys': Math.random() * 10 + 10
-        })
-    }
-}
+  var number =
+    level *
+    parseInt(
+      (20 * core.bigmap.width * core.bigmap.height) /
+        (core._WIDTH_ * core._HEIGHT_)
+    );
+  for (var a = 0; a < number; a++) {
+    core.animateFrame.weather.nodes.push({
+      x: Math.random() * core.bigmap.width * 32,
+      y: Math.random() * core.bigmap.height * 32,
+      l: Math.random() * 2.5,
+      xs: -4 + Math.random() * 4 + 2,
+      ys: Math.random() * 10 + 10,
+    });
+  }
+};
 
 control.prototype._weather_snow = function (level) {
-    var number = level * parseInt(20 * core.bigmap.width * core.bigmap.height / (core._WIDTH_ * core._HEIGHT_));
-    for (var a = 0; a < number; a++) {
-        core.animateFrame.weather.nodes.push({
-            'x': Math.random() * core.bigmap.width * 32,
-            'y': Math.random() * core.bigmap.height * 32,
-            'r': Math.random() * 5 + 1,
-            'd': Math.random() * Math.min(level, 200),
-        })
-    }
-}
+  var number =
+    level *
+    parseInt(
+      (20 * core.bigmap.width * core.bigmap.height) /
+        (core._WIDTH_ * core._HEIGHT_)
+    );
+  for (var a = 0; a < number; a++) {
+    core.animateFrame.weather.nodes.push({
+      x: Math.random() * core.bigmap.width * 32,
+      y: Math.random() * core.bigmap.height * 32,
+      r: Math.random() * 5 + 1,
+      d: Math.random() * Math.min(level, 200),
+    });
+  }
+};
 
 control.prototype._weather_fog = function (level) {
-    if (!core.animateFrame.weather.fog) return;
-    core.animateFrame.weather.nodes = [{
-        'image': core.animateFrame.weather.fog,
-        'level': 40 * level,
-        'x': 0,
-        'y': -core._PY_ / 2,
-        'dx': -Math.random() * 1.5,
-        'dy': Math.random(),
-        'delta': 0.001,
-    }];
-}
+  if (!core.animateFrame.weather.fog) return;
+  core.animateFrame.weather.nodes = [
+    {
+      image: core.animateFrame.weather.fog,
+      level: 40 * level,
+      x: 0,
+      y: -core._PY_ / 2,
+      dx: -Math.random() * 1.5,
+      dy: Math.random(),
+      delta: 0.001,
+    },
+  ];
+};
 
 control.prototype._weather_cloud = function (level) {
-    if (!core.animateFrame.weather.cloud) return;
-    core.animateFrame.weather.nodes = [{
-        'image': core.animateFrame.weather.cloud,
-        'level': 40 * level,
-        'x': 0,
-        'y': -core._PY_ / 2,
-        'dx': -Math.random() * 1.5,
-        'dy': Math.random(),
-        'delta': 0.001,
-    }];
-}
+  if (!core.animateFrame.weather.cloud) return;
+  core.animateFrame.weather.nodes = [
+    {
+      image: core.animateFrame.weather.cloud,
+      level: 40 * level,
+      x: 0,
+      y: -core._PY_ / 2,
+      dx: -Math.random() * 1.5,
+      dy: Math.random(),
+      delta: 0.001,
+    },
+  ];
+};
 
 control.prototype._weather_sun = function (level) {
-    if (!core.animateFrame.weather.sun) return;
-    // 直接绘制
-    core.clearMap('weather');
-    core.drawImage(
-        'weather', core.animateFrame.weather.sun, 0, 0, core.animateFrame.weather.sun.width, core.animateFrame.weather.sun.height, 0, 0, core._PX_, core._PY_
-    );
-    core.setOpacity('weather', level / 10);
-    core.animateFrame.weather.nodes = [{ opacity: level / 10, delta: 0.01 }];
-}
+  if (!core.animateFrame.weather.sun) return;
+  // 直接绘制
+  core.clearMap("weather");
+  core.drawImage(
+    "weather",
+    core.animateFrame.weather.sun,
+    0,
+    0,
+    core.animateFrame.weather.sun.width,
+    core.animateFrame.weather.sun.height,
+    0,
+    0,
+    core._PX_,
+    core._PY_
+  );
+  core.setOpacity("weather", level / 10);
+  core.animateFrame.weather.nodes = [{ opacity: level / 10, delta: 0.01 }];
+};
 
 ////// 更改画面色调 //////
 control.prototype.setCurtain = function (color, time, moveMode, callback) {
-    if (time == null) time = 750;
-    if (time <= 0) time = 0;
-    if (!core.status.curtainColor)
-        core.status.curtainColor = [0, 0, 0, 0];
-    if (!color) color = [0, 0, 0, 0];
-    if (color[3] == null) color[3] = 1;
-    color[3] = core.clamp(color[3], 0, 1);
+  if (time == null) time = 750;
+  if (time <= 0) time = 0;
+  if (!core.status.curtainColor) core.status.curtainColor = [0, 0, 0, 0];
+  if (!color) color = [0, 0, 0, 0];
+  if (color[3] == null) color[3] = 1;
+  color[3] = core.clamp(color[3], 0, 1);
 
-    if (time == 0) {
-        // 直接变色
-        core.clearMap('curtain');
-        core.fillRect('curtain', 0, 0, core._PX_, core._PY_, core.arrayToRGBA(color));
-        core.status.curtainColor = color;
-        if (callback) callback();
-        return;
+  if (time == 0) {
+    // 直接变色
+    core.clearMap("curtain");
+    core.fillRect(
+      "curtain",
+      0,
+      0,
+      core._PX_,
+      core._PY_,
+      core.arrayToRGBA(color)
+    );
+    core.status.curtainColor = color;
+    if (callback) callback();
+    return;
+  }
+
+  this._setCurtain_animate(
+    core.status.curtainColor,
+    color,
+    time,
+    moveMode,
+    callback
+  );
+};
+
+control.prototype._setCurtain_animate = function (
+  nowColor,
+  color,
+  time,
+  moveMode,
+  callback
+) {
+  time /= Math.max(core.status.replay.speed, 1);
+  var per_time = 10,
+    step = 0,
+    steps = parseInt(time / per_time);
+  if (steps <= 0) steps = 1;
+  var curr = nowColor;
+  var moveFunc = core.applyEasing(moveMode);
+
+  var cb = function () {
+    core.status.curtainColor = curr;
+    if (callback) callback();
+  };
+  var animate = setInterval(function () {
+    step++;
+    curr = [
+      nowColor[0] + (color[0] - nowColor[0]) * moveFunc(step / steps),
+      nowColor[1] + (color[1] - nowColor[1]) * moveFunc(step / steps),
+      nowColor[2] + (color[2] - nowColor[2]) * moveFunc(step / steps),
+      nowColor[3] + (color[3] - nowColor[3]) * moveFunc(step / steps),
+    ];
+    core.clearMap("curtain");
+    core.fillRect(
+      "curtain",
+      0,
+      0,
+      core._PX_,
+      core._PY_,
+      core.arrayToRGBA(curr)
+    );
+    if (step == steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      cb();
     }
+  }, per_time);
 
-    this._setCurtain_animate(core.status.curtainColor, color, time, moveMode, callback);
-}
-
-control.prototype._setCurtain_animate = function (nowColor, color, time, moveMode, callback) {
-    time /= Math.max(core.status.replay.speed, 1)
-    var per_time = 10, step = 0, steps = parseInt(time / per_time);
-    if (steps <= 0) steps = 1;
-    var curr = nowColor;
-    var moveFunc = core.applyEasing(moveMode);
-
-    var cb = function () {
-        core.status.curtainColor = curr;
-        if (callback) callback();
-    }
-    var animate = setInterval(function () {
-        step++;
-        curr = [
-            nowColor[0] + (color[0] - nowColor[0]) * moveFunc(step / steps),
-            nowColor[1] + (color[1] - nowColor[1]) * moveFunc(step / steps),
-            nowColor[2] + (color[2] - nowColor[2]) * moveFunc(step / steps),
-            nowColor[3] + (color[3] - nowColor[3]) * moveFunc(step / steps),
-        ]
-        core.clearMap('curtain');
-        core.fillRect('curtain', 0, 0, core._PX_, core._PY_, core.arrayToRGBA(curr));
-        if (step == steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            cb();
-        }
-    }, per_time);
-
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = cb;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = cb;
+};
 
 ////// 画面闪烁 //////
-control.prototype.screenFlash = function (color, time, times, moveMode, callback) {
-    times = times || 1;
-    time = time / 3;
-    var nowColor = core.clone(core.status.curtainColor);
-    core.setCurtain(color, time, moveMode, function () {
-        core.setCurtain(nowColor, time * 2, moveMode, function () {
-            if (times > 1)
-                core.screenFlash(color, time * 3, times - 1, moveMode, callback);
-            else {
-                if (callback) callback();
-            }
-        });
+control.prototype.screenFlash = function (
+  color,
+  time,
+  times,
+  moveMode,
+  callback
+) {
+  times = times || 1;
+  time = time / 3;
+  var nowColor = core.clone(core.status.curtainColor);
+  core.setCurtain(color, time, moveMode, function () {
+    core.setCurtain(nowColor, time * 2, moveMode, function () {
+      if (times > 1)
+        core.screenFlash(color, time * 3, times - 1, moveMode, callback);
+      else {
+        if (callback) callback();
+      }
     });
-}
+  });
+};
 
 ////// 播放背景音乐 //////
 control.prototype.playBgm = function (bgm, startTime) {
-    bgm = core.getMappedName(bgm);
-    if (main.mode != 'play' || !core.material.bgms[bgm]) return;
-    // 如果不允许播放
-    if (!core.musicStatus.bgmStatus) {
-        try {
-            core.musicStatus.playingBgm = bgm;
-            core.musicStatus.lastBgm = bgm;
-            core.material.bgms[bgm].pause();
-        }
-        catch (e) {
-            console.error(e);
-        }
-        return;
-    }
-    this.setMusicBtn();
-
+  bgm = core.getMappedName(bgm);
+  if (main.mode != "play" || !core.material.bgms[bgm]) return;
+  // 如果不允许播放
+  if (!core.musicStatus.bgmStatus) {
     try {
-        this._playBgm_play(bgm, startTime);
+      core.musicStatus.playingBgm = bgm;
+      core.musicStatus.lastBgm = bgm;
+      core.material.bgms[bgm].pause();
+    } catch (e) {
+      console.error(e);
     }
-    catch (e) {
-        console.log("无法播放BGM " + bgm);
-        console.error(e);
-        core.musicStatus.playingBgm = null;
-    }
-}
+    return;
+  }
+  this.setMusicBtn();
+
+  try {
+    this._playBgm_play(bgm, startTime);
+  } catch (e) {
+    console.log("无法播放BGM " + bgm);
+    console.error(e);
+    core.musicStatus.playingBgm = null;
+  }
+};
 
 control.prototype._playBgm_play = function (bgm, startTime) {
-    // 如果当前正在播放,且和本BGM相同,直接忽略
-    if (core.musicStatus.playingBgm == bgm && !core.material.bgms[core.musicStatus.playingBgm].paused) {
-        return;
-    }
-    // 如果正在播放中,暂停
-    if (core.musicStatus.playingBgm) {
-        core.material.bgms[core.musicStatus.playingBgm].pause();
-    }
-    // 缓存BGM
-    core.loader.loadBgm(bgm);
-    // 播放当前BGM
-    core.material.bgms[bgm].volume = core.musicStatus.userVolume * core.musicStatus.designVolume;
-    core.material.bgms[bgm].currentTime = startTime || 0;
-    core.material.bgms[bgm].play();
-    core.musicStatus.playingBgm = bgm;
-    core.musicStatus.lastBgm = bgm;
-    core.setBgmSpeed(100);
-}
+  // 如果当前正在播放,且和本BGM相同,直接忽略
+  if (
+    core.musicStatus.playingBgm == bgm &&
+    !core.material.bgms[core.musicStatus.playingBgm].paused
+  ) {
+    return;
+  }
+  // 如果正在播放中,暂停
+  if (core.musicStatus.playingBgm) {
+    core.material.bgms[core.musicStatus.playingBgm].pause();
+  }
+  // 缓存BGM
+  core.loader.loadBgm(bgm);
+  // 播放当前BGM
+  core.material.bgms[bgm].volume =
+    core.musicStatus.userVolume * core.musicStatus.designVolume;
+  core.material.bgms[bgm].currentTime = startTime || 0;
+  core.material.bgms[bgm].play();
+  core.musicStatus.playingBgm = bgm;
+  core.musicStatus.lastBgm = bgm;
+  core.setBgmSpeed(100);
+};
 
 ///// 设置当前背景音乐的播放速度 //////
 control.prototype.setBgmSpeed = function (speed, usePitch) {
-    var bgm = core.musicStatus.playingBgm;
-    if (main.mode != 'play' || !core.material.bgms[bgm]) return;
-    bgm = core.material.bgms[bgm];
-    if (speed < 30 || speed > 300) return;
-    bgm.playbackRate = speed / 100;
-    core.musicStatus.bgmSpeed = speed;
+  var bgm = core.musicStatus.playingBgm;
+  if (main.mode != "play" || !core.material.bgms[bgm]) return;
+  bgm = core.material.bgms[bgm];
+  if (speed < 30 || speed > 300) return;
+  bgm.playbackRate = speed / 100;
+  core.musicStatus.bgmSpeed = speed;
 
-    if (bgm.preservesPitch != null) {
-        if (bgm.__preservesPitch == null) bgm.__preservesPitch = bgm.preservesPitch;
-        if (usePitch == null) bgm.preservesPitch = bgm.__preservesPitch;
-        else if (usePitch) bgm.preservesPitch = false;
-        else bgm.preservesPitch = true;
-        core.musicStatus.bgmUsePitch = usePitch;
-    }
-}
+  if (bgm.preservesPitch != null) {
+    if (bgm.__preservesPitch == null) bgm.__preservesPitch = bgm.preservesPitch;
+    if (usePitch == null) bgm.preservesPitch = bgm.__preservesPitch;
+    else if (usePitch) bgm.preservesPitch = false;
+    else bgm.preservesPitch = true;
+    core.musicStatus.bgmUsePitch = usePitch;
+  }
+};
 
 ////// 暂停背景音乐的播放 //////
 control.prototype.pauseBgm = function () {
-    if (main.mode != 'play') return;
-    try {
-        if (core.musicStatus.playingBgm) {
-            core.musicStatus.pauseTime = core.material.bgms[core.musicStatus.playingBgm].currentTime;
-            core.material.bgms[core.musicStatus.playingBgm].pause();
-            core.musicStatus.playingBgm = null;
-        }
+  if (main.mode != "play") return;
+  try {
+    if (core.musicStatus.playingBgm) {
+      core.musicStatus.pauseTime =
+        core.material.bgms[core.musicStatus.playingBgm].currentTime;
+      core.material.bgms[core.musicStatus.playingBgm].pause();
+      core.musicStatus.playingBgm = null;
     }
-    catch (e) {
-        console.log("无法暂停BGM");
-        console.error(e);
-    }
-    this.setMusicBtn();
-}
+  } catch (e) {
+    console.log("无法暂停BGM");
+    console.error(e);
+  }
+  this.setMusicBtn();
+};
 
 ////// 恢复背景音乐的播放 //////
 control.prototype.resumeBgm = function (resumeTime) {
-    if (main.mode != 'play') return;
-    try {
-        var speed = core.musicStatus.bgmSpeed;
-        var usePitch = core.musicStatus.bgmUsePitch;
-        core.playBgm(core.musicStatus.playingBgm || core.musicStatus.lastBgm || main.startBgm,
-            resumeTime ? core.musicStatus.pauseTime : 0);
-        if (resumeTime) {
-            core.setBgmSpeed(speed, usePitch);
-        }
+  if (main.mode != "play") return;
+  try {
+    var speed = core.musicStatus.bgmSpeed;
+    var usePitch = core.musicStatus.bgmUsePitch;
+    core.playBgm(
+      core.musicStatus.playingBgm || core.musicStatus.lastBgm || main.startBgm,
+      resumeTime ? core.musicStatus.pauseTime : 0
+    );
+    if (resumeTime) {
+      core.setBgmSpeed(speed, usePitch);
     }
-    catch (e) {
-        console.log("无法恢复BGM");
-        console.error(e);
-    }
-    this.setMusicBtn();
-}
+  } catch (e) {
+    console.log("无法恢复BGM");
+    console.error(e);
+  }
+  this.setMusicBtn();
+};
 
 control.prototype.setMusicBtn = function () {
-    if (core.musicStatus.bgmStatus)
-        core.dom.musicBtn.src = "";
-    else
-        core.dom.musicBtn.src = "";
-}
+  if (core.musicStatus.bgmStatus)
+    core.dom.musicBtn.src =
+      "";
+  else
+    core.dom.musicBtn.src =
+      "";
+};
 
 ////// 更改背景音乐的播放 //////
 control.prototype.triggerBgm = function () {
-    if (main.mode != 'play') return;
+  if (main.mode != "play") return;
 
-    core.musicStatus.bgmStatus = !core.musicStatus.bgmStatus;
-    if (core.musicStatus.bgmStatus)
-        this.resumeBgm();
-    else
-        this.pauseBgm();
-    core.setLocalStorage('bgmStatus', core.musicStatus.bgmStatus);
-}
+  core.musicStatus.bgmStatus = !core.musicStatus.bgmStatus;
+  if (core.musicStatus.bgmStatus) this.resumeBgm();
+  else this.pauseBgm();
+  core.setLocalStorage("bgmStatus", core.musicStatus.bgmStatus);
+};
 
 ////// 播放音频 //////
 control.prototype.playSound = function (sound, pitch, callback) {
-    sound = core.getMappedName(sound);
-    if (main.mode != 'play' || !core.musicStatus.soundStatus || !core.material.sounds[sound]) return;
-    try {
-        if (core.musicStatus.audioContext != null) {
-            var source = core.musicStatus.audioContext.createBufferSource();
-            source.__name = sound;
-            source.buffer = core.material.sounds[sound];
-            source.connect(core.musicStatus.gainNode);
-            var id = setTimeout(null);
-            if (pitch && pitch >= 30 && pitch <= 300) {
-                source.playbackRate.setValueAtTime(pitch / 100, 0);
-            }
-            source.onended = function () {
-                delete core.musicStatus.playingSounds[id];
-                if (callback) callback();
-            }
-            core.musicStatus.playingSounds[id] = source;
-            if (source.start) source.start(0);
-            else if (source.noteOn) source.noteOn(0);
-            return id;
-        }
-        else {
-            core.material.sounds[sound].volume = core.musicStatus.userVolume;
-            core.material.sounds[sound].play();
-            if (callback) callback();
-        }
+  sound = core.getMappedName(sound);
+  if (
+    main.mode != "play" ||
+    !core.musicStatus.soundStatus ||
+    !core.material.sounds[sound]
+  )
+    return;
+  try {
+    if (core.musicStatus.audioContext != null) {
+      var source = core.musicStatus.audioContext.createBufferSource();
+      source.__name = sound;
+      source.buffer = core.material.sounds[sound];
+      source.connect(core.musicStatus.gainNode);
+      var id = setTimeout(null);
+      if (pitch && pitch >= 30 && pitch <= 300) {
+        source.playbackRate.setValueAtTime(pitch / 100, 0);
+      }
+      source.onended = function () {
+        delete core.musicStatus.playingSounds[id];
+        if (callback) callback();
+      };
+      core.musicStatus.playingSounds[id] = source;
+      if (source.start) source.start(0);
+      else if (source.noteOn) source.noteOn(0);
+      return id;
+    } else {
+      core.material.sounds[sound].volume = core.musicStatus.userVolume;
+      core.material.sounds[sound].play();
+      if (callback) callback();
     }
-    catch (e) {
-        console.log("无法播放SE " + sound);
-        console.error(e);
-    }
-}
+  } catch (e) {
+    console.log("无法播放SE " + sound);
+    console.error(e);
+  }
+};
 
 ////// 停止所有音频 //////
 control.prototype.stopSound = function (id) {
-    if (id == null) {
-        Object.keys(core.musicStatus.playingSounds).forEach(function (id) {
-            core.control.stopSound(id);
-        });
-        return;
-    }
-    var source = core.musicStatus.playingSounds[id];
-    if (!source) return;
-    try {
-        if (source.stop) source.stop();
-        else if (source.noteOff) source.noteOff();
-    }
-    catch (e) {
-        console.error(e);
-    }
-    delete core.musicStatus.playingSounds[id];
-}
+  if (id == null) {
+    Object.keys(core.musicStatus.playingSounds).forEach(function (id) {
+      core.control.stopSound(id);
+    });
+    return;
+  }
+  var source = core.musicStatus.playingSounds[id];
+  if (!source) return;
+  try {
+    if (source.stop) source.stop();
+    else if (source.noteOff) source.noteOff();
+  } catch (e) {
+    console.error(e);
+  }
+  delete core.musicStatus.playingSounds[id];
+};
 
 ////// 获得当前正在播放的所有(指定)音效的id列表 //////
 control.prototype.getPlayingSounds = function (name) {
-    name = core.getMappedName(name);
-    return Object.keys(core.musicStatus.playingSounds).filter(function (one) {
-        return name == null || core.musicStatus.playingSounds[one].__name == name
-    });
-}
+  name = core.getMappedName(name);
+  return Object.keys(core.musicStatus.playingSounds).filter(function (one) {
+    return name == null || core.musicStatus.playingSounds[one].__name == name;
+  });
+};
 
 ////// 检查bgm状态 //////
 control.prototype.checkBgm = function () {
-    core.playBgm(core.musicStatus.playingBgm || main.startBgm);
-}
+  core.playBgm(core.musicStatus.playingBgm || main.startBgm);
+};
 
 ///// 设置屏幕放缩 //////
 control.prototype.setDisplayScale = function (delta) {
-    var index = core.domStyle.availableScale.indexOf(core.domStyle.scale);
-    if (index < 0) return;
-    index = (index + delta + core.domStyle.availableScale.length) % core.domStyle.availableScale.length;
-    core.domStyle.scale = core.domStyle.availableScale[index];
-    core.setLocalStorage('scale', core.domStyle.scale);
-    core.resize();
-}
+  var index = core.domStyle.availableScale.indexOf(core.domStyle.scale);
+  if (index < 0) return;
+  index =
+    (index + delta + core.domStyle.availableScale.length) %
+    core.domStyle.availableScale.length;
+  core.domStyle.scale = core.domStyle.availableScale[index];
+  core.setLocalStorage("scale", core.domStyle.scale);
+  core.resize();
+};
 
 // ------ 状态栏,工具栏等相关 ------ //
 
 ////// 清空状态栏 //////
 control.prototype.clearStatusBar = function () {
-    Object.keys(core.statusBar).forEach(function (e) {
-        if (core.statusBar[e].innerHTML != null) {
-            core.statusBar[e].innerHTML = "&nbsp;";
-            core.statusBar[e].removeAttribute('_style');
-            core.statusBar[e].removeAttribute('_value');
-        }
-    })
-    core.statusBar.image.book.style.opacity = 0.3;
-    if (!core.flags.equipboxButton)
-        core.statusBar.image.fly.style.opacity = 0.3;
-}
+  Object.keys(core.statusBar).forEach(function (e) {
+    if (core.statusBar[e].innerHTML != null) {
+      core.statusBar[e].innerHTML = "&nbsp;";
+      core.statusBar[e].removeAttribute("_style");
+      core.statusBar[e].removeAttribute("_value");
+    }
+  });
+  core.statusBar.image.book.style.opacity = 0.3;
+  if (!core.flags.equipboxButton) core.statusBar.image.fly.style.opacity = 0.3;
+};
 
 ////// 更新状态栏 //////
 control.prototype.updateStatusBar = function (doNotCheckAutoEvents, immediate) {
-    if (!core.isPlaying()) return;
-    if (immediate) {
-        return this.updateStatusBar_update();
-    }
-    if (!doNotCheckAutoEvents) this.noAutoEvents = false;
-    if (core.isReplaying()) return this.updateStatusBar_update();
-    if (!core.control.updateNextFrame) {
-        core.control.updateNextFrame = true;
-        requestAnimationFrame(this.updateStatusBar_update);
-    }
+  if (!core.isPlaying()) return;
+  if (immediate) {
+    return this.updateStatusBar_update();
+  }
+  if (!doNotCheckAutoEvents) this.noAutoEvents = false;
+  if (core.isReplaying()) return this.updateStatusBar_update();
+  if (!core.control.updateNextFrame) {
+    core.control.updateNextFrame = true;
+    requestAnimationFrame(this.updateStatusBar_update);
+  }
 };
 
 control.prototype.updateStatusBar_update = function () {
-    core.control.updateNextFrame = false;
-    if (!core.isPlaying() || core.hasFlag('__statistics__')) return;
-    core.control.controldata.updateStatusBar();
-    if (!core.control.noAutoEvents) core.checkAutoEvents();
-    core.control._updateStatusBar_setToolboxIcon();
-    core.clearRouteFolding();
-    core.control.noAutoEvents = true;
+  core.control.updateNextFrame = false;
+  if (!core.isPlaying() || core.hasFlag("__statistics__")) return;
+  core.control.controldata.updateStatusBar();
+  if (!core.control.noAutoEvents) core.checkAutoEvents();
+  core.control._updateStatusBar_setToolboxIcon();
+  core.clearRouteFolding();
+  core.control.noAutoEvents = true;
 };
 
 control.prototype._updateStatusBar_setToolboxIcon = function () {
-    if (core.isReplaying()) {
-        core.statusBar.image.book.src = core.status.replay.pausing ? core.statusBar.icons.play.src : core.statusBar.icons.pause.src;
-        core.statusBar.image.book.style.opacity = 1;
-        core.statusBar.image.fly.src = core.statusBar.icons.stop.src;
-        core.statusBar.image.fly.style.opacity = 1;
-        core.statusBar.image.toolbox.src = core.statusBar.icons.rewind.src;
-        core.statusBar.image.keyboard.src = core.statusBar.icons.book.src;
-        core.statusBar.image.shop.src = core.statusBar.icons.floor.src;
-        core.statusBar.image.save.src = core.statusBar.icons.speedDown.src;
-        core.statusBar.image.save.style.opacity = 1;
-        core.statusBar.image.load.src = core.statusBar.icons.speedUp.src;
-        core.statusBar.image.settings.src = core.statusBar.icons.save.src;
+  if (core.isReplaying()) {
+    core.statusBar.image.book.src = core.status.replay.pausing
+      ? core.statusBar.icons.play.src
+      : core.statusBar.icons.pause.src;
+    core.statusBar.image.book.style.opacity = 1;
+    core.statusBar.image.fly.src = core.statusBar.icons.stop.src;
+    core.statusBar.image.fly.style.opacity = 1;
+    core.statusBar.image.toolbox.src = core.statusBar.icons.rewind.src;
+    core.statusBar.image.keyboard.src = core.statusBar.icons.book.src;
+    core.statusBar.image.shop.src = core.statusBar.icons.floor.src;
+    core.statusBar.image.save.src = core.statusBar.icons.speedDown.src;
+    core.statusBar.image.save.style.opacity = 1;
+    core.statusBar.image.load.src = core.statusBar.icons.speedUp.src;
+    core.statusBar.image.settings.src = core.statusBar.icons.save.src;
+  } else {
+    core.statusBar.image.book.src = core.statusBar.icons.book.src;
+    core.statusBar.image.book.style.opacity = core.hasItem("book") ? 1 : 0.3;
+    if (!core.flags.equipboxButton) {
+      core.statusBar.image.fly.src = core.statusBar.icons.fly.src;
+      core.statusBar.image.fly.style.opacity = core.hasItem("fly") ? 1 : 0.3;
+    } else {
+      core.statusBar.image.fly.src = core.statusBar.icons.equipbox.src;
+      core.statusBar.image.fly.style.opacity = 1;
     }
-    else {
-        core.statusBar.image.book.src = core.statusBar.icons.book.src;
-        core.statusBar.image.book.style.opacity = core.hasItem('book') ? 1 : 0.3;
-        if (!core.flags.equipboxButton) {
-            core.statusBar.image.fly.src = core.statusBar.icons.fly.src;
-            core.statusBar.image.fly.style.opacity = core.hasItem('fly') ? 1 : 0.3;
-        }
-        else {
-            core.statusBar.image.fly.src = core.statusBar.icons.equipbox.src;
-            core.statusBar.image.fly.style.opacity = 1;
-        }
-        core.statusBar.image.toolbox.src = core.statusBar.icons.toolbox.src;
-        core.statusBar.image.keyboard.src = core.statusBar.icons.keyboard.src;
-        core.statusBar.image.shop.src = core.statusBar.icons.shop.src;
-        core.statusBar.image.save.src = core.statusBar.icons.save.src;
-        core.statusBar.image.save.style.opacity = core.hasFlag('__forbidSave__') ? 0.3 : 1;
-        core.statusBar.image.load.src = core.statusBar.icons.load.src;
-        core.statusBar.image.settings.src = core.statusBar.icons.settings.src;
-    }
-}
+    core.statusBar.image.toolbox.src = core.statusBar.icons.toolbox.src;
+    core.statusBar.image.keyboard.src = core.statusBar.icons.keyboard.src;
+    core.statusBar.image.shop.src = core.statusBar.icons.shop.src;
+    core.statusBar.image.save.src = core.statusBar.icons.save.src;
+    core.statusBar.image.save.style.opacity = core.hasFlag("__forbidSave__")
+      ? 0.3
+      : 1;
+    core.statusBar.image.load.src = core.statusBar.icons.load.src;
+    core.statusBar.image.settings.src = core.statusBar.icons.settings.src;
+  }
+};
 
 control.prototype.showStatusBar = function () {
-    if (main.mode == 'editor') return;
-    if (core.domStyle.showStatusBar) return;
-    var statusItems = core.dom.status;
-    core.domStyle.showStatusBar = true;
-    core.removeFlag('hideStatusBar');
-    // 显示
-    for (var i = 0; i < statusItems.length; ++i)
-        statusItems[i].style.opacity = 1;
-    this.setToolbarButton(false);
-    core.dom.tools.hard.style.display = 'block';
-    core.dom.toolBar.style.display = 'block';
-}
+  if (main.mode == "editor") return;
+  if (core.domStyle.showStatusBar) return;
+  var statusItems = core.dom.status;
+  core.domStyle.showStatusBar = true;
+  core.removeFlag("hideStatusBar");
+  // 显示
+  for (var i = 0; i < statusItems.length; ++i) statusItems[i].style.opacity = 1;
+  this.setToolbarButton(false);
+  core.dom.tools.hard.style.display = "block";
+  core.dom.toolBar.style.display = "block";
+};
 
 control.prototype.hideStatusBar = function (showToolbox) {
-    if (main.mode == 'editor') return;
+  if (main.mode == "editor") return;
 
-    // 如果原本就是隐藏的,则先显示
-    if (!core.domStyle.showStatusBar)
-        this.showStatusBar();
-    if (core.isReplaying()) showToolbox = true;
+  // 如果原本就是隐藏的,则先显示
+  if (!core.domStyle.showStatusBar) this.showStatusBar();
+  if (core.isReplaying()) showToolbox = true;
 
-    var statusItems = core.dom.status, toolItems = core.dom.tools;
-    core.domStyle.showStatusBar = false;
-    core.setFlag('hideStatusBar', true);
-    core.setFlag('showToolbox', showToolbox || null);
-    // 隐藏
-    for (var i = 0; i < statusItems.length; ++i)
-        statusItems[i].style.opacity = 0;
-    if ((!core.domStyle.isVertical && !core.flags.extendToolbar) || !showToolbox) {
-        for (var i = 0; i < toolItems.length; ++i)
-            toolItems[i].style.display = 'none';
-    }
-    if (!core.domStyle.isVertical && !core.flags.extendToolbar) {
-        core.dom.toolBar.style.display = 'none';
-    }
-}
+  var statusItems = core.dom.status,
+    toolItems = core.dom.tools;
+  core.domStyle.showStatusBar = false;
+  core.setFlag("hideStatusBar", true);
+  core.setFlag("showToolbox", showToolbox || null);
+  // 隐藏
+  for (var i = 0; i < statusItems.length; ++i) statusItems[i].style.opacity = 0;
+  if (
+    (!core.domStyle.isVertical && !core.flags.extendToolbar) ||
+    !showToolbox
+  ) {
+    for (var i = 0; i < toolItems.length; ++i)
+      toolItems[i].style.display = "none";
+  }
+  if (!core.domStyle.isVertical && !core.flags.extendToolbar) {
+    core.dom.toolBar.style.display = "none";
+  }
+};
 
 ////// 更新状态栏的勇士图标 //////
 control.prototype.updateHeroIcon = function (name) {
-    name = name || "hero.png";
-    if (core.statusBar.icons.name == name) return;
-    core.statusBar.icons.name = name;
+  name = name || "hero.png";
+  if (core.statusBar.icons.name == name) return;
+  core.statusBar.icons.name = name;
 
-    var image = core.material.images.hero;
-    // 全身图
-    var w = core.material.icons.hero.width || 32;
-    var h = core.material.icons.hero.height || 48;
-    var ratio = Math.min(w / h, 1), width = 32 * ratio, left = 16 - width / 2;
+  var image = core.material.images.hero;
+  // 全身图
+  var w = core.material.icons.hero.width || 32;
+  var h = core.material.icons.hero.height || 48;
+  var ratio = Math.min(w / h, 1),
+    width = 32 * ratio,
+    left = 16 - width / 2;
 
-    var canvas = document.createElement("canvas");
-    var ctx = canvas.getContext("2d");
-    canvas.width = 32;
-    canvas.height = 32;
-    core.drawImage(ctx, image, 0, 0, w, h, left, 0, width, 32);
+  var canvas = document.createElement("canvas");
+  var ctx = canvas.getContext("2d");
+  canvas.width = 32;
+  canvas.height = 32;
+  core.drawImage(ctx, image, 0, 0, w, h, left, 0, width, 32);
 
-    core.statusBar.image.name.src = canvas.toDataURL("image/png");
-}
+  core.statusBar.image.name.src = canvas.toDataURL("image/png");
+};
 
 ////// 改变工具栏为按钮1-8 //////
 control.prototype.setToolbarButton = function (useButton) {
-    if (!core.domStyle.showStatusBar) {
-        // 隐藏状态栏时检查竖屏
-        if (!core.domStyle.isVertical && !core.flags.extendToolbar) {
-            for (var i = 0; i < core.dom.tools.length; ++i)
-                core.dom.tools[i].style.display = 'none';
-            return;
-        }
-        if (!core.hasFlag('showToolbox')) return;
-        else core.dom.tools.hard.style.display = 'block';
+  if (!core.domStyle.showStatusBar) {
+    // 隐藏状态栏时检查竖屏
+    if (!core.domStyle.isVertical && !core.flags.extendToolbar) {
+      for (var i = 0; i < core.dom.tools.length; ++i)
+        core.dom.tools[i].style.display = "none";
+      return;
     }
+    if (!core.hasFlag("showToolbox")) return;
+    else core.dom.tools.hard.style.display = "block";
+  }
 
-    if (useButton == null) useButton = core.domStyle.toolbarBtn;
-    if ((!core.domStyle.isVertical && !core.flags.extendToolbar)) useButton = false;
-    core.domStyle.toolbarBtn = useButton;
+  if (useButton == null) useButton = core.domStyle.toolbarBtn;
+  if (!core.domStyle.isVertical && !core.flags.extendToolbar) useButton = false;
+  core.domStyle.toolbarBtn = useButton;
 
-    if (useButton) {
-        ["book", "fly", "toolbox", "keyboard", "shop", "save", "load", "settings"].forEach(function (t) {
-            core.statusBar.image[t].style.display = 'none';
-        });
-        ["btn1", "btn2", "btn3", "btn4", "btn5", "btn6", "btn7", "btn8"].forEach(function (t) {
-            core.statusBar.image[t].style.display = 'block';
-        })
-        main.statusBar.image.btn8.style.filter = core.getLocalStorage('altKey') ? 'sepia(1) contrast(1.5)' : '';
-    }
-    else {
-        ["btn1", "btn2", "btn3", "btn4", "btn5", "btn6", "btn7", "btn8"].forEach(function (t) {
-            core.statusBar.image[t].style.display = 'none';
-        });
-        ["book", "fly", "toolbox", "save", "load", "settings"].forEach(function (t) {
-            core.statusBar.image[t].style.display = 'block';
-        });
-        core.statusBar.image.keyboard.style.display
-            = core.statusBar.image.shop.style.display
-            = core.domStyle.isVertical || core.flags.extendToolbar ? "block" : "none";
-    }
-}
+  if (useButton) {
+    [
+      "book",
+      "fly",
+      "toolbox",
+      "keyboard",
+      "shop",
+      "save",
+      "load",
+      "settings",
+    ].forEach(function (t) {
+      core.statusBar.image[t].style.display = "none";
+    });
+    ["btn1", "btn2", "btn3", "btn4", "btn5", "btn6", "btn7", "btn8"].forEach(
+      function (t) {
+        core.statusBar.image[t].style.display = "block";
+      }
+    );
+    main.statusBar.image.btn8.style.filter = core.getLocalStorage("altKey")
+      ? "sepia(1) contrast(1.5)"
+      : "";
+  } else {
+    ["btn1", "btn2", "btn3", "btn4", "btn5", "btn6", "btn7", "btn8"].forEach(
+      function (t) {
+        core.statusBar.image[t].style.display = "none";
+      }
+    );
+    ["book", "fly", "toolbox", "save", "load", "settings"].forEach(function (
+      t
+    ) {
+      core.statusBar.image[t].style.display = "block";
+    });
+    core.statusBar.image.keyboard.style.display =
+      core.statusBar.image.shop.style.display =
+        core.domStyle.isVertical || core.flags.extendToolbar ? "block" : "none";
+  }
+};
 
 ////// ------ resize处理 ------ //
 
 control.prototype._shouldDisplayStatus = function (id) {
-    if (id == null) {
-        var toDraw = [], status = core.dom.status;
-        for (var i = 0; i < status.length; ++i) {
-            var dom = core.dom.status[i], idCol = dom.id;
-            if (idCol.indexOf("Col") != idCol.length - 3) continue;
-            var id = idCol.substring(0, idCol.length - 3);
-            if (!this._shouldDisplayStatus(id)) continue;
-            toDraw.push(id);
-        }
-        return toDraw;
+  if (id == null) {
+    var toDraw = [],
+      status = core.dom.status;
+    for (var i = 0; i < status.length; ++i) {
+      var dom = core.dom.status[i],
+        idCol = dom.id;
+      if (idCol.indexOf("Col") != idCol.length - 3) continue;
+      var id = idCol.substring(0, idCol.length - 3);
+      if (!this._shouldDisplayStatus(id)) continue;
+      toDraw.push(id);
     }
-    var obj = {};
-    core.flags.statusBarItems.forEach(function (v) { obj[v] = true; })
-    switch (id) {
-        case 'floor': return obj.enableFloor;
-        case 'name': return obj.enableName;
-        case 'lv': return obj.enableLv;
-        case 'hp': return obj.enableHP;
-        case 'hpmax': return obj.enableHPMax;
-        case 'mana': return obj.enableMana;
-        case 'atk': return obj.enableAtk;
-        case 'def': return obj.enableDef;
-        case 'mdef': return obj.enableMDef;
-        case 'money': return obj.enableMoney;
-        case 'exp': return obj.enableExp && !obj.levelUpLeftMode;
-        case 'up': return obj.enableLevelUp;
-        case 'skill': return obj.enableSkill;
-        case 'key': return obj.enableKeys;
-        case 'pzf': return obj.enablePZF;
-        case 'debuff': return obj.enableDebuff;
-        default: return true;
-    }
-}
+    return toDraw;
+  }
+  var obj = {};
+  core.flags.statusBarItems.forEach(function (v) {
+    obj[v] = true;
+  });
+  switch (id) {
+    case "floor":
+      return obj.enableFloor;
+    case "name":
+      return obj.enableName;
+    case "lv":
+      return obj.enableLv;
+    case "hp":
+      return obj.enableHP;
+    case "hpmax":
+      return obj.enableHPMax;
+    case "mana":
+      return obj.enableMana;
+    case "atk":
+      return obj.enableAtk;
+    case "def":
+      return obj.enableDef;
+    case "mdef":
+      return obj.enableMDef;
+    case "money":
+      return obj.enableMoney;
+    case "exp":
+      return obj.enableExp && !obj.levelUpLeftMode;
+    case "up":
+      return obj.enableLevelUp;
+    case "skill":
+      return obj.enableSkill;
+    case "key":
+      return obj.enableKeys;
+    case "pzf":
+      return obj.enablePZF;
+    case "debuff":
+      return obj.enableDebuff;
+    default:
+      return true;
+  }
+};
 
 ////// 注册一个resize函数 //////
 // name为名称,可供注销使用
 // func可以是一个函数,或者是插件中的函数名;可以接受obj参数,详见resize函数。
 control.prototype.registerResize = function (name, func) {
-    this.unregisterResize(name);
-    this.resizes.push({ name: name, func: func });
-}
+  this.unregisterResize(name);
+  this.resizes.push({ name: name, func: func });
+};
 
 ////// 注销一个resize函数 //////
 control.prototype.unregisterResize = function (name) {
-    this.resizes = this.resizes.filter(function (b) { return b.name != name; });
-}
+  this.resizes = this.resizes.filter(function (b) {
+    return b.name != name;
+  });
+};
 
 control.prototype._doResize = function (obj) {
-    for (var i in this.resizes) {
-        try {
-            if (core.doFunc(this.resizes[i].func, this, obj)) return true;
-        } catch (e) {
-            console.error(e);
-            console.error("ERROR in resizes[" + this.resizes[i].name + "]:已自动注销该项。");
-            this.unregisterResize(this.resizes[i].name);
-        }
+  for (var i in this.resizes) {
+    try {
+      if (core.doFunc(this.resizes[i].func, this, obj)) return true;
+    } catch (e) {
+      console.error(e);
+      console.error(
+        "ERROR in resizes[" + this.resizes[i].name + "]:已自动注销该项。"
+      );
+      this.unregisterResize(this.resizes[i].name);
     }
-    return false;
-}
+  }
+  return false;
+};
 
 ////// 屏幕分辨率改变后重新自适应 //////
 control.prototype.resize = function () {
-    if (main.mode == 'editor') return;
-    var clientWidth = main.dom.body.clientWidth, clientHeight = main.dom.body.clientHeight;
-    var BORDER = 3;
-    var extendToolbar = core.flags.extendToolbar;
-    let hideLeftStatusBar = core.flags.hideLeftStatusBar;
-    var BAR_WIDTH = hideLeftStatusBar ? 0 : Math.round(core._PY_ * 0.31);
+  if (main.mode == "editor") return;
+  var clientWidth = main.dom.body.clientWidth,
+    clientHeight = main.dom.body.clientHeight;
+  var BORDER = 3;
+  var extendToolbar = core.flags.extendToolbar;
+  let hideLeftStatusBar = core.flags.hideLeftStatusBar;
+  var BAR_WIDTH = hideLeftStatusBar ? 0 : Math.round(core._PY_ * 0.31);
 
-    var horizontalMaxRatio = (clientHeight - 2 * BORDER - (hideLeftStatusBar ? BORDER : 0)) / (core._PY_ + (hideLeftStatusBar ? 38 : 0));
+  var horizontalMaxRatio =
+    (clientHeight - 2 * BORDER - (hideLeftStatusBar ? BORDER : 0)) /
+    (core._PY_ + (hideLeftStatusBar ? 38 : 0));
 
-    if (clientWidth - 3 * BORDER >= core._PX_ + BAR_WIDTH || (clientWidth > clientHeight && horizontalMaxRatio < 1)) {
-        // 横屏
-        core.domStyle.isVertical = false;
+  if (
+    clientWidth - 3 * BORDER >= core._PX_ + BAR_WIDTH ||
+    (clientWidth > clientHeight && horizontalMaxRatio < 1)
+  ) {
+    // 横屏
+    core.domStyle.isVertical = false;
 
-        core.domStyle.availableScale = [];
-        [1, 1.25, 1.5, 1.75, 2, 2.25, 2.5].forEach(function (v) {
-            if (clientWidth - 3 * BORDER >= v * (core._PX_ + BAR_WIDTH) && horizontalMaxRatio >= v) {
-                core.domStyle.availableScale.push(v);
-            }
-        });
-        if (core.domStyle.availableScale.indexOf(core.domStyle.scale) < 0) {
-            core.domStyle.scale = Math.min(1, horizontalMaxRatio);
-        }
-    }
-    else {
-        // 竖屏
-        core.domStyle.isVertical = true;
-        core.domStyle.scale = Math.min((clientWidth - 2 * BORDER) / core._PX_);
-        core.domStyle.availableScale = [];
-        extendToolbar = false;
-        hideLeftStatusBar = false;
-        BAR_WIDTH = Math.round(core._PX_ * 0.3);
+    core.domStyle.availableScale = [];
+    [1, 1.25, 1.5, 1.75, 2, 2.25, 2.5].forEach(function (v) {
+      if (
+        clientWidth - 3 * BORDER >= v * (core._PX_ + BAR_WIDTH) &&
+        horizontalMaxRatio >= v
+      ) {
+        core.domStyle.availableScale.push(v);
+      }
+    });
+    if (core.domStyle.availableScale.indexOf(core.domStyle.scale) < 0) {
+      core.domStyle.scale = Math.min(1, horizontalMaxRatio);
     }
+  } else {
+    // 竖屏
+    core.domStyle.isVertical = true;
+    core.domStyle.scale = Math.min((clientWidth - 2 * BORDER) / core._PX_);
+    core.domStyle.availableScale = [];
+    extendToolbar = false;
+    hideLeftStatusBar = false;
+    BAR_WIDTH = Math.round(core._PX_ * 0.3);
+  }
 
-    var statusDisplayArr = this._shouldDisplayStatus(), count = statusDisplayArr.length;
-    var statusCanvas = core.flags.statusCanvas, statusCanvasRows = core.values.statusCanvasRowsOnMobile || 3;
-    var col = statusCanvas ? statusCanvasRows : Math.ceil(count / 3);
-    if (col > 5) {
-        if (statusCanvas) alert("自绘状态栏的在竖屏下的行数应不超过5!");
-        else alert("当前状态栏数目(" + count + ")大于15,请调整到不超过15以避免手机端出现显示问题。");
-    }
-    var globalAttribute = core.status.globalAttribute || core.initStatus.globalAttribute;
+  var statusDisplayArr = this._shouldDisplayStatus(),
+    count = statusDisplayArr.length;
+  var statusCanvas = core.flags.statusCanvas,
+    statusCanvasRows = core.values.statusCanvasRowsOnMobile || 3;
+  var col = statusCanvas ? statusCanvasRows : Math.ceil(count / 3);
+  if (col > 5) {
+    if (statusCanvas) alert("自绘状态栏的在竖屏下的行数应不超过5!");
+    else
+      alert(
+        "当前状态栏数目(" +
+          count +
+          ")大于15,请调整到不超过15以避免手机端出现显示问题。"
+      );
+  }
+  var globalAttribute =
+    core.status.globalAttribute || core.initStatus.globalAttribute;
 
-    var obj = {
-        clientWidth: clientWidth,
-        clientHeight: clientHeight,
-        BORDER: BORDER,
-        BAR_WIDTH: BAR_WIDTH,
-        TOOLBAR_HEIGHT: 38,
-        outerWidth: core._PX_ * core.domStyle.scale + 2 * BORDER,
-        outerHeight: core._PY_ * core.domStyle.scale + 2 * BORDER,
-        globalAttribute: globalAttribute,
-        border: '3px ' + core.arrayToRGBA(globalAttribute.borderColor) + ' solid',
-        statusDisplayArr: statusDisplayArr,
-        count: count,
-        col: col,
-        statusBarHeightInVertical: core.domStyle.isVertical ? (32 * col + 6) * core.domStyle.scale + 2 * BORDER : 0,
-        toolbarHeightInVertical: core.domStyle.isVertical ? 38 * core.domStyle.scale + 2 * BORDER : 0,
-        extendToolbar: extendToolbar,
-        is15x15: false,
-        hideLeftStatusBar
-    };
+  var obj = {
+    clientWidth: clientWidth,
+    clientHeight: clientHeight,
+    BORDER: BORDER,
+    BAR_WIDTH: BAR_WIDTH,
+    TOOLBAR_HEIGHT: 38,
+    outerWidth: core._PX_ * core.domStyle.scale + 2 * BORDER,
+    outerHeight: core._PY_ * core.domStyle.scale + 2 * BORDER,
+    globalAttribute: globalAttribute,
+    border: "3px " + core.arrayToRGBA(globalAttribute.borderColor) + " solid",
+    statusDisplayArr: statusDisplayArr,
+    count: count,
+    col: col,
+    statusBarHeightInVertical: core.domStyle.isVertical
+      ? (32 * col + 6) * core.domStyle.scale + 2 * BORDER
+      : 0,
+    toolbarHeightInVertical: core.domStyle.isVertical
+      ? 38 * core.domStyle.scale + 2 * BORDER
+      : 0,
+    extendToolbar: extendToolbar,
+    is15x15: false,
+    hideLeftStatusBar,
+  };
 
-    this._doResize(obj);
-    this.setToolbarButton();
-    core.updateStatusBar();
-}
+  this._doResize(obj);
+  this.setToolbarButton();
+  core.updateStatusBar();
+};
 
 control.prototype._resize_gameGroup = function (obj) {
-    var startBackground = core.domStyle.isVertical ? (main.styles.startVerticalBackground || main.styles.startBackground) : main.styles.startBackground;
-    if (main.dom.startBackground.getAttribute('__src__') != startBackground) {
-        main.dom.startBackground.setAttribute('__src__', startBackground);
-        main.dom.startBackground.src = startBackground;
-    }
+  var startBackground = core.domStyle.isVertical
+    ? main.styles.startVerticalBackground || main.styles.startBackground
+    : main.styles.startBackground;
+  if (main.dom.startBackground.getAttribute("__src__") != startBackground) {
+    main.dom.startBackground.setAttribute("__src__", startBackground);
+    main.dom.startBackground.src = startBackground;
+  }
 
-    var gameGroup = core.dom.gameGroup;
-    var totalWidth, totalHeight;
-    if (core.domStyle.isVertical) {
-        totalWidth = obj.outerWidth;
-        totalHeight = obj.outerHeight + obj.statusBarHeightInVertical + obj.toolbarHeightInVertical
-    }
-    else {
-        totalWidth = obj.outerWidth + obj.BAR_WIDTH * core.domStyle.scale + (obj.hideLeftStatusBar ? 0 : obj.BORDER);
-        totalHeight = obj.outerHeight + (obj.extendToolbar ? obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER : 0);
-    }
-    gameGroup.style.width = totalWidth + "px";
-    gameGroup.style.height = totalHeight + "px";
-    gameGroup.style.left = (obj.clientWidth - totalWidth) / 2 + "px";
-    gameGroup.style.top = (obj.clientHeight - totalHeight) / 2 + "px";
-    // floorMsgGroup
-    var floorMsgGroup = core.dom.floorMsgGroup;
-    floorMsgGroup.style = obj.globalAttribute.floorChangingStyle;
-    floorMsgGroup.style.width = obj.outerWidth - 2 * obj.BORDER + "px";
-    floorMsgGroup.style.height = totalHeight - 2 * obj.BORDER + "px";
-    floorMsgGroup.style.fontSize = 16 * core.domStyle.scale + "px";
-    // startPanel
-    core.dom.startPanel.style.fontSize = 16 * core.domStyle.scale + "px";
-    // musicBtn
-    if (core.domStyle.isVertical || core.domStyle.scale < 1) {
-        core.dom.musicBtn.style.right = core.dom.musicBtn.style.bottom = "3px";
-    }
-    else {
-        core.dom.musicBtn.style.right = (obj.clientWidth - totalWidth) / 2 + "px";
-        core.dom.musicBtn.style.bottom = (obj.clientHeight - totalHeight) / 2 - 27 + "px";
-    }
-}
+  var gameGroup = core.dom.gameGroup;
+  var totalWidth, totalHeight;
+  if (core.domStyle.isVertical) {
+    totalWidth = obj.outerWidth;
+    totalHeight =
+      obj.outerHeight +
+      obj.statusBarHeightInVertical +
+      obj.toolbarHeightInVertical;
+  } else {
+    totalWidth =
+      obj.outerWidth +
+      obj.BAR_WIDTH * core.domStyle.scale +
+      (obj.hideLeftStatusBar ? 0 : obj.BORDER);
+    totalHeight =
+      obj.outerHeight +
+      (obj.extendToolbar
+        ? obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER
+        : 0);
+  }
+  gameGroup.style.width = totalWidth + "px";
+  gameGroup.style.height = totalHeight + "px";
+  gameGroup.style.left = (obj.clientWidth - totalWidth) / 2 + "px";
+  gameGroup.style.top = (obj.clientHeight - totalHeight) / 2 + "px";
+  // floorMsgGroup
+  var floorMsgGroup = core.dom.floorMsgGroup;
+  floorMsgGroup.style = obj.globalAttribute.floorChangingStyle;
+  floorMsgGroup.style.width = obj.outerWidth - 2 * obj.BORDER + "px";
+  floorMsgGroup.style.height = totalHeight - 2 * obj.BORDER + "px";
+  floorMsgGroup.style.fontSize = 16 * core.domStyle.scale + "px";
+  // startPanel
+  core.dom.startPanel.style.fontSize = 16 * core.domStyle.scale + "px";
+  // musicBtn
+  if (core.domStyle.isVertical || core.domStyle.scale < 1) {
+    core.dom.musicBtn.style.right = core.dom.musicBtn.style.bottom = "3px";
+  } else {
+    core.dom.musicBtn.style.right = (obj.clientWidth - totalWidth) / 2 + "px";
+    core.dom.musicBtn.style.bottom =
+      (obj.clientHeight - totalHeight) / 2 - 27 + "px";
+  }
+};
 
 control.prototype._resize_canvas = function (obj) {
-    var innerWidth = (core._PX_ * core.domStyle.scale) + "px", innerHeight = (core._PY_ * core.domStyle.scale) + "px";
-    if (!core.isPlaying()) {
-        for (var i = 0; i < core.dom.gameCanvas.length; ++i) {
-            var ctx = core.dom.gameCanvas[i].getContext('2d');
-            core.resizeCanvas(ctx, core._PX_, core._PY_);
-            core.dom.gameCanvas[i].style.width = innerWidth;
-            core.dom.gameCanvas[i].style.height = innerHeight;
-        }
-    } else {
-        requestAnimationFrame(function () {
-            for (var i = 0; i < core.dom.gameCanvas.length; ++i) {
-                core.dom.gameCanvas[i].style.width = innerWidth;
-                core.dom.gameCanvas[i].style.height = innerHeight;
-            }
-        });
+  var innerWidth = core._PX_ * core.domStyle.scale + "px",
+    innerHeight = core._PY_ * core.domStyle.scale + "px";
+  if (!core.isPlaying()) {
+    for (var i = 0; i < core.dom.gameCanvas.length; ++i) {
+      var ctx = core.dom.gameCanvas[i].getContext("2d");
+      core.resizeCanvas(ctx, core._PX_, core._PY_);
+      core.dom.gameCanvas[i].style.width = innerWidth;
+      core.dom.gameCanvas[i].style.height = innerHeight;
     }
-    core.dom.gif.style.width = innerWidth;
-    core.dom.gif.style.height = innerHeight;
-    core.dom.gif2.style.width = innerWidth;
-    core.dom.gif2.style.height = innerHeight;
-    core.dom.gameDraw.style.width = innerWidth;
-    core.dom.gameDraw.style.height = innerHeight;
-    core.dom.gameDraw.style.top = obj.statusBarHeightInVertical + "px";
-    core.dom.gameDraw.style.right = 0;
-    core.dom.gameDraw.style.border = obj.border;
-    // resize bigmap
-    core.bigmap.canvas.forEach(function (cn) {
-        var ratio = core.canvas[cn].canvas.hasAttribute('isHD') ? core.domStyle.ratio : 1;
-        core.canvas[cn].canvas.style.width = core.canvas[cn].canvas.width / ratio * core.domStyle.scale + "px";
-        core.canvas[cn].canvas.style.height = core.canvas[cn].canvas.height / ratio * core.domStyle.scale + "px";
+  } else {
+    requestAnimationFrame(function () {
+      for (var i = 0; i < core.dom.gameCanvas.length; ++i) {
+        core.dom.gameCanvas[i].style.width = innerWidth;
+        core.dom.gameCanvas[i].style.height = innerHeight;
+      }
     });
-    // resize dynamic canvas
-    if (!core.isPlaying()) {
-        for (var name in core.dymCanvas) {
-            var ctx = core.dymCanvas[name], canvas = ctx.canvas;
-            // core.maps._setHDCanvasSize(ctx, parseFloat(canvas.getAttribute('_width')), parseFloat(canvas.getAttribute('_height')));
-            canvas.style.left = parseFloat(canvas.getAttribute("_left")) * core.domStyle.scale + "px";
-            canvas.style.top = parseFloat(canvas.getAttribute("_top")) * core.domStyle.scale + "px";
-            var scale = canvas.getAttribute('_scale') || 1;
-            core.resizeCanvas(canvas, canvas.width * scale / core.domStyle.scale, canvas.height * scale / core.domStyle.scale);
-        }
-    } else {
-        for (var name in core.dymCanvas) {
-            var ctx = core.dymCanvas[name], canvas = ctx.canvas;
-            core.resizeCanvas(ctx, parseFloat(canvas.getAttribute("_width")), parseFloat(canvas.getAttribute("_height")))
-            canvas.style.left = parseFloat(canvas.getAttribute("_left")) * core.domStyle.scale + "px";
-            canvas.style.top = parseFloat(canvas.getAttribute("_top")) * core.domStyle.scale + "px";
-        }
+  }
+  core.dom.gif.style.width = innerWidth;
+  core.dom.gif.style.height = innerHeight;
+  core.dom.gif2.style.width = innerWidth;
+  core.dom.gif2.style.height = innerHeight;
+  core.dom.gameDraw.style.width = innerWidth;
+  core.dom.gameDraw.style.height = innerHeight;
+  core.dom.gameDraw.style.top = obj.statusBarHeightInVertical + "px";
+  core.dom.gameDraw.style.right = 0;
+  core.dom.gameDraw.style.border = obj.border;
+  // resize bigmap
+  core.bigmap.canvas.forEach(function (cn) {
+    var ratio = core.canvas[cn].canvas.hasAttribute("isHD")
+      ? core.domStyle.ratio
+      : 1;
+    core.canvas[cn].canvas.style.width =
+      (core.canvas[cn].canvas.width / ratio) * core.domStyle.scale + "px";
+    core.canvas[cn].canvas.style.height =
+      (core.canvas[cn].canvas.height / ratio) * core.domStyle.scale + "px";
+  });
+  // resize dynamic canvas
+  if (!core.isPlaying()) {
+    for (var name in core.dymCanvas) {
+      var ctx = core.dymCanvas[name],
+        canvas = ctx.canvas;
+      // core.maps._setHDCanvasSize(ctx, parseFloat(canvas.getAttribute('_width')), parseFloat(canvas.getAttribute('_height')));
+      canvas.style.left =
+        parseFloat(canvas.getAttribute("_left")) * core.domStyle.scale + "px";
+      canvas.style.top =
+        parseFloat(canvas.getAttribute("_top")) * core.domStyle.scale + "px";
+      var scale = canvas.getAttribute("_scale") || 1;
+      core.resizeCanvas(
+        canvas,
+        (canvas.width * scale) / core.domStyle.scale,
+        (canvas.height * scale) / core.domStyle.scale
+      );
     }
-    // resize next
-    main.dom.next.style.width = main.dom.next.style.height = 5 * core.domStyle.scale + "px";
-    main.dom.next.style.borderBottomWidth = main.dom.next.style.borderRightWidth = 4 * core.domStyle.scale + "px";
-}
+  } else {
+    for (var name in core.dymCanvas) {
+      var ctx = core.dymCanvas[name],
+        canvas = ctx.canvas;
+      core.resizeCanvas(
+        ctx,
+        parseFloat(canvas.getAttribute("_width")),
+        parseFloat(canvas.getAttribute("_height"))
+      );
+      canvas.style.left =
+        parseFloat(canvas.getAttribute("_left")) * core.domStyle.scale + "px";
+      canvas.style.top =
+        parseFloat(canvas.getAttribute("_top")) * core.domStyle.scale + "px";
+    }
+  }
+  // resize next
+  main.dom.next.style.width = main.dom.next.style.height =
+    5 * core.domStyle.scale + "px";
+  main.dom.next.style.borderBottomWidth = main.dom.next.style.borderRightWidth =
+    4 * core.domStyle.scale + "px";
+};
 
 control.prototype._resize_statusBar = function (obj) {
-    // statusBar
-    var statusBar = core.dom.statusBar;
-    if (core.domStyle.isVertical) {
-        statusBar.style.width = obj.outerWidth + "px";
-        statusBar.style.height = obj.statusBarHeightInVertical + "px";
-        statusBar.style.background = obj.globalAttribute.statusTopBackground;
-        statusBar.style.fontSize = 16 * core.domStyle.scale + "px";
+  // statusBar
+  var statusBar = core.dom.statusBar;
+  if (core.domStyle.isVertical) {
+    statusBar.style.width = obj.outerWidth + "px";
+    statusBar.style.height = obj.statusBarHeightInVertical + "px";
+    statusBar.style.background = obj.globalAttribute.statusTopBackground;
+    statusBar.style.fontSize = 16 * core.domStyle.scale + "px";
+  } else {
+    statusBar.style.width =
+      obj.BAR_WIDTH * core.domStyle.scale + obj.BORDER + "px";
+    statusBar.style.height =
+      obj.outerHeight +
+      (obj.extendToolbar
+        ? obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER
+        : 0) +
+      "px";
+    statusBar.style.background = obj.globalAttribute.statusLeftBackground;
+    // --- 计算文字大小
+    if (obj.hideLeftStatusBar) {
+      statusBar.style.fontSize = 16 * core.domStyle.scale + "px";
+    } else {
+      statusBar.style.fontSize =
+        16 *
+          Math.min(1, (core._HEIGHT_ - 4) / obj.count) *
+          core.domStyle.scale +
+        "px";
     }
-    else {
-        statusBar.style.width = (obj.BAR_WIDTH * core.domStyle.scale + obj.BORDER) + "px";
-        statusBar.style.height = obj.outerHeight + (obj.extendToolbar ? obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER : 0) + "px";
-        statusBar.style.background = obj.globalAttribute.statusLeftBackground;
-        // --- 计算文字大小
-        if (obj.hideLeftStatusBar) {
-            statusBar.style.fontSize = 16 * core.domStyle.scale + "px";
-        } else {
-            statusBar.style.fontSize = 16 * Math.min(1, (core._HEIGHT_ - 4) / obj.count) * core.domStyle.scale + "px";
-        }
-    }
-    statusBar.style.display = obj.hideLeftStatusBar ? 'none' : 'block';
-    statusBar.style.borderTop = statusBar.style.borderLeft = obj.border;
-    statusBar.style.borderRight = core.domStyle.isVertical ? obj.border : '';
-    statusBar.style.borderBottom = core.domStyle.isVertical ? '' : obj.border;
-    // 自绘状态栏
-    if (core.domStyle.isVertical) {
-        core.dom.statusCanvas.style.width = core._PX_ * core.domStyle.scale + "px";
-        core.dom.statusCanvas.style.height = obj.statusBarHeightInVertical - 3 + "px";
-        core.maps._setHDCanvasSize(core.dom.statusCanvasCtx, core._PX_, obj.col * 32 + 9);
-    }
-    else {
-        core.dom.statusCanvas.style.width = obj.BAR_WIDTH * core.domStyle.scale + "px";
-        core.dom.statusCanvas.style.height = obj.outerHeight - 2 * obj.BORDER + (obj.extendToolbar ? obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER : 0) + "px";
-        core.maps._setHDCanvasSize(core.dom.statusCanvasCtx, obj.BAR_WIDTH, core._PY_ + (obj.extendToolbar ? obj.TOOLBAR_HEIGHT + obj.BORDER : 0));
-    }
-    core.dom.statusCanvas.style.display = core.flags.statusCanvas && !obj.hideLeftStatusBar ? "block" : "none";
-}
+  }
+  statusBar.style.display = obj.hideLeftStatusBar ? "none" : "block";
+  statusBar.style.borderTop = statusBar.style.borderLeft = obj.border;
+  statusBar.style.borderRight = core.domStyle.isVertical ? obj.border : "";
+  statusBar.style.borderBottom = core.domStyle.isVertical ? "" : obj.border;
+  // 自绘状态栏
+  if (core.domStyle.isVertical) {
+    core.dom.statusCanvas.style.width = core._PX_ * core.domStyle.scale + "px";
+    core.dom.statusCanvas.style.height =
+      obj.statusBarHeightInVertical - 3 + "px";
+    core.maps._setHDCanvasSize(
+      core.dom.statusCanvasCtx,
+      core._PX_,
+      obj.col * 32 + 9
+    );
+  } else {
+    core.dom.statusCanvas.style.width =
+      obj.BAR_WIDTH * core.domStyle.scale + "px";
+    core.dom.statusCanvas.style.height =
+      obj.outerHeight -
+      2 * obj.BORDER +
+      (obj.extendToolbar
+        ? obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER
+        : 0) +
+      "px";
+    core.maps._setHDCanvasSize(
+      core.dom.statusCanvasCtx,
+      obj.BAR_WIDTH,
+      core._PY_ + (obj.extendToolbar ? obj.TOOLBAR_HEIGHT + obj.BORDER : 0)
+    );
+  }
+  core.dom.statusCanvas.style.display =
+    core.flags.statusCanvas && !obj.hideLeftStatusBar ? "block" : "none";
+};
 
 control.prototype._resize_status = function (obj) {
-    var statusHeight;
-    if (core.domStyle.isVertical) {
-        statusHeight = 32 * core.domStyle.scale * 0.8;
-    } else {
-        statusHeight = (obj.hideLeftStatusBar ? core._HEIGHT_ : core._HEIGHT_ - 4) / obj.count * 32 * core.domStyle.scale * 0.8;
-    }
-    // status
-    for (var i = 0; i < core.dom.status.length; ++i) {
-        var id = core.dom.status[i].id, style = core.dom.status[i].style;
-        if (id.endsWith("Col")) id = id.substring(0, id.length - 3);
-        style.display = core.flags.statusCanvas || obj.statusDisplayArr.indexOf(id) < 0 ? 'none' : 'block';
-        style.margin = 3 * core.domStyle.scale + "px";
-        style.height = statusHeight + "px";
-        style.maxWidth = obj.BAR_WIDTH * core.domStyle.scale * (core.domStyle.isVertical ? 0.95 : 1) + obj.BORDER + "px";
-        if (obj.is15x15 && !core.domStyle.isVertical)
-            style.marginLeft = 11 * core.domStyle.scale + "px";
-    }
-    // statusLabels, statusTexts
-    for (var i = 0; i < core.dom.statusLabels.length; ++i) {
-        core.dom.statusLabels[i].style.lineHeight = statusHeight + "px";
-        core.dom.statusLabels[i].style.marginLeft = 6 * core.domStyle.scale + "px";
-    }
-    for (var i = 0; i < core.dom.statusTexts.length; ++i) {
-        core.dom.statusTexts[i].style.color = core.arrayToRGBA(obj.globalAttribute.statusBarColor);
-    }
-    // keys
-    if (core.flags.statusBarItems.indexOf('enableGreenKey') >= 0) {
-        core.dom.keyCol.style.fontSize = '0.75em';
-        core.statusBar.greenKey.style.display = '';
-    } else {
-        core.dom.keyCol.style.fontSize = '';
-        core.statusBar.greenKey.style.display = 'none';
-    }
-}
+  var statusHeight;
+  if (core.domStyle.isVertical) {
+    statusHeight = 32 * core.domStyle.scale * 0.8;
+  } else {
+    statusHeight =
+      ((obj.hideLeftStatusBar ? core._HEIGHT_ : core._HEIGHT_ - 4) /
+        obj.count) *
+      32 *
+      core.domStyle.scale *
+      0.8;
+  }
+  // status
+  for (var i = 0; i < core.dom.status.length; ++i) {
+    var id = core.dom.status[i].id,
+      style = core.dom.status[i].style;
+    if (id.endsWith("Col")) id = id.substring(0, id.length - 3);
+    style.display =
+      core.flags.statusCanvas || obj.statusDisplayArr.indexOf(id) < 0
+        ? "none"
+        : "block";
+    style.margin = 3 * core.domStyle.scale + "px";
+    style.height = statusHeight + "px";
+    style.maxWidth =
+      obj.BAR_WIDTH *
+        core.domStyle.scale *
+        (core.domStyle.isVertical ? 0.95 : 1) +
+      obj.BORDER +
+      "px";
+    if (obj.is15x15 && !core.domStyle.isVertical)
+      style.marginLeft = 11 * core.domStyle.scale + "px";
+  }
+  // statusLabels, statusTexts
+  for (var i = 0; i < core.dom.statusLabels.length; ++i) {
+    core.dom.statusLabels[i].style.lineHeight = statusHeight + "px";
+    core.dom.statusLabels[i].style.marginLeft = 6 * core.domStyle.scale + "px";
+  }
+  for (var i = 0; i < core.dom.statusTexts.length; ++i) {
+    core.dom.statusTexts[i].style.color = core.arrayToRGBA(
+      obj.globalAttribute.statusBarColor
+    );
+  }
+  // keys
+  if (core.flags.statusBarItems.indexOf("enableGreenKey") >= 0) {
+    core.dom.keyCol.style.fontSize = "0.75em";
+    core.statusBar.greenKey.style.display = "";
+  } else {
+    core.dom.keyCol.style.fontSize = "";
+    core.statusBar.greenKey.style.display = "none";
+  }
+};
 
 control.prototype._resize_toolBar = function (obj) {
-    // toolBar
-    var toolBar = core.dom.toolBar;
-    if (core.domStyle.isVertical) {
-        toolBar.style.left = 0;
-        toolBar.style.right = "";
-        toolBar.style.width = obj.outerWidth + "px";
-        toolBar.style.top = obj.statusBarHeightInVertical + obj.outerHeight + "px";
-        toolBar.style.height = obj.toolbarHeightInVertical + "px";
-        toolBar.style.background = obj.globalAttribute.toolsBackground;
-    }
-    else {
-        if (obj.extendToolbar || obj.hideLeftStatusBar) {
-            toolBar.style.left = "";
-            toolBar.style.right = 0;
-            toolBar.style.width = obj.outerWidth + "px";
-            toolBar.style.top = obj.outerHeight + "px";
-            toolBar.style.height = obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER + "px";
-            toolBar.style.background = obj.globalAttribute.toolsBackground;
-        } else {
-            toolBar.style.left = 0;
-            toolBar.style.right = "";
-            toolBar.style.width = obj.BAR_WIDTH * core.domStyle.scale + obj.BORDER + "px";
-            toolBar.style.top = 0.75 * obj.outerHeight + "px";
-            toolBar.style.height = 0.25 * obj.outerHeight + "px";
-            toolBar.style.background = 'transparent';
-        }
-    }
-    toolBar.style.borderLeft = obj.border;
-    toolBar.style.borderRight = toolBar.style.borderBottom = core.domStyle.isVertical || obj.extendToolbar ? obj.border : '';
-    toolBar.style.fontSize = 16 * core.domStyle.scale + "px";
-
-    if (!core.domStyle.showStatusBar && !core.domStyle.isVertical && !obj.extendToolbar) {
-        toolBar.style.display = 'none';
+  // toolBar
+  var toolBar = core.dom.toolBar;
+  if (core.domStyle.isVertical) {
+    toolBar.style.left = 0;
+    toolBar.style.right = "";
+    toolBar.style.width = obj.outerWidth + "px";
+    toolBar.style.top = obj.statusBarHeightInVertical + obj.outerHeight + "px";
+    toolBar.style.height = obj.toolbarHeightInVertical + "px";
+    toolBar.style.background = obj.globalAttribute.toolsBackground;
+  } else {
+    if (obj.extendToolbar || obj.hideLeftStatusBar) {
+      toolBar.style.left = "";
+      toolBar.style.right = 0;
+      toolBar.style.width = obj.outerWidth + "px";
+      toolBar.style.top = obj.outerHeight + "px";
+      toolBar.style.height =
+        obj.TOOLBAR_HEIGHT * core.domStyle.scale + obj.BORDER + "px";
+      toolBar.style.background = obj.globalAttribute.toolsBackground;
     } else {
-        toolBar.style.display = 'block';
+      toolBar.style.left = 0;
+      toolBar.style.right = "";
+      toolBar.style.width =
+        obj.BAR_WIDTH * core.domStyle.scale + obj.BORDER + "px";
+      toolBar.style.top = 0.75 * obj.outerHeight + "px";
+      toolBar.style.height = 0.25 * obj.outerHeight + "px";
+      toolBar.style.background = "transparent";
     }
-}
+  }
+  toolBar.style.borderLeft = obj.border;
+  toolBar.style.borderRight = toolBar.style.borderBottom =
+    core.domStyle.isVertical || obj.extendToolbar ? obj.border : "";
+  toolBar.style.fontSize = 16 * core.domStyle.scale + "px";
+
+  if (
+    !core.domStyle.showStatusBar &&
+    !core.domStyle.isVertical &&
+    !obj.extendToolbar
+  ) {
+    toolBar.style.display = "none";
+  } else {
+    toolBar.style.display = "block";
+  }
+};
 
 control.prototype._resize_tools = function (obj) {
-    var toolsHeight = 32 * core.domStyle.scale * ((core.domStyle.isVertical || obj.extendToolbar) && !obj.is15x15 ? 0.95 : 1);
-    var toolsMarginLeft;
-    if (core.domStyle.isVertical || obj.extendToolbar)
-        toolsMarginLeft = (core._HALF_WIDTH_ - 3) * 3 * core.domStyle.scale;
-    else
-        toolsMarginLeft = (obj.BAR_WIDTH * core.domStyle.scale - 9 - toolsHeight * 3) / 4;
-    for (var i = 0; i < core.dom.tools.length; ++i) {
-        var style = core.dom.tools[i].style;
-        style.height = toolsHeight + "px";
-        style.marginLeft = toolsMarginLeft + "px";
-        style.marginTop = 3 * core.domStyle.scale + "px"
-    }
-    core.dom.hard.style.lineHeight = toolsHeight + "px";
-    if (core.domStyle.isVertical || obj.extendToolbar) {
-        core.dom.hard.style.width = obj.outerWidth - 9 * toolsMarginLeft - 8.5 * toolsHeight - 12 + "px";
-    }
-    else {
-        core.dom.hard.style.width = obj.BAR_WIDTH * core.domStyle.scale - 9 - 2 * toolsMarginLeft + "px";
-        if (!obj.is15x15) core.dom.hard.style.marginTop = 0;
-    }
-}
+  var toolsHeight =
+    32 *
+    core.domStyle.scale *
+    ((core.domStyle.isVertical || obj.extendToolbar) && !obj.is15x15
+      ? 0.95
+      : 1);
+  var toolsMarginLeft;
+  if (core.domStyle.isVertical || obj.extendToolbar)
+    toolsMarginLeft = (core._HALF_WIDTH_ - 3) * 3 * core.domStyle.scale;
+  else
+    toolsMarginLeft =
+      (obj.BAR_WIDTH * core.domStyle.scale - 9 - toolsHeight * 3) / 4;
+  for (var i = 0; i < core.dom.tools.length; ++i) {
+    var style = core.dom.tools[i].style;
+    style.height = toolsHeight + "px";
+    style.marginLeft = toolsMarginLeft + "px";
+    style.marginTop = 3 * core.domStyle.scale + "px";
+  }
+  core.dom.hard.style.lineHeight = toolsHeight + "px";
+  if (core.domStyle.isVertical || obj.extendToolbar) {
+    core.dom.hard.style.width =
+      obj.outerWidth - 9 * toolsMarginLeft - 8.5 * toolsHeight - 12 + "px";
+  } else {
+    core.dom.hard.style.width =
+      obj.BAR_WIDTH * core.domStyle.scale - 9 - 2 * toolsMarginLeft + "px";
+    if (!obj.is15x15) core.dom.hard.style.marginTop = 0;
+  }
+};
diff --git a/libs/events.js b/libs/events.js
index 66d9954..3aabd20 100644
--- a/libs/events.js
+++ b/libs/events.js
@@ -1,429 +1,485 @@
-
 "use strict";
 
-function events () {
-    this._init();
+function events() {
+  this._init();
 }
 
 ////// 初始化 //////
 events.prototype._init = function () {
-    this.eventdata = functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a.events;
-    this.commonEvent = events_c12a15a8_c380_4b28_8144_256cba95f760.commonEvent;
-    this.systemEvents = {};
-    this.actions = {};
-}
+  this.eventdata = functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a.events;
+  this.commonEvent = events_c12a15a8_c380_4b28_8144_256cba95f760.commonEvent;
+  this.systemEvents = {};
+  this.actions = {};
+};
 
 // ------ 初始化,开始和结束 ------ //
 
 /// 初始化游戏
 events.prototype.resetGame = function (hero, hard, floorId, maps, values) {
-    this.eventdata.resetGame(hero, hard, floorId, maps, values);
-}
+  this.eventdata.resetGame(hero, hard, floorId, maps, values);
+};
 
 ////// 游戏开始事件 //////
 events.prototype.startGame = function (hard, seed, route, callback) {
-    main.dom.levelChooseButtons.style.display = 'none';
-    main.dom.startButtonGroup.style.display = 'none';
-    hard = hard || "";
+  main.dom.levelChooseButtons.style.display = "none";
+  main.dom.startButtonGroup.style.display = "none";
+  hard = hard || "";
 
-    if (main.mode != 'play') return;
+  if (main.mode != "play") return;
 
-    // 无动画的开始游戏
-    if (core.flags.startUsingCanvas || route != null) {
-        core.dom.startPanel.style.display = 'none';
-        this._startGame_start(hard, seed, route, callback);
-    }
-    else {
-        core.hideStartAnimate(function () {
-            core.events._startGame_start(hard, seed, route, callback);
-        });
-    }
-}
+  // 无动画的开始游戏
+  if (core.flags.startUsingCanvas || route != null) {
+    core.dom.startPanel.style.display = "none";
+    this._startGame_start(hard, seed, route, callback);
+  } else {
+    core.hideStartAnimate(function () {
+      core.events._startGame_start(hard, seed, route, callback);
+    });
+  }
+};
 
 events.prototype._startGame_start = function (hard, seed, route, callback) {
-    core.resetGame(core.firstData.hero, hard, null, core.cloneArray(core.initStatus.maps));
-    core.setHeroLoc('x', -1);
-    core.setHeroLoc('y', -1);
+  core.resetGame(
+    core.firstData.hero,
+    hard,
+    null,
+    core.cloneArray(core.initStatus.maps)
+  );
+  core.setHeroLoc("x", -1);
+  core.setHeroLoc("y", -1);
 
-    if (seed != null && seed > 0) {
-        core.setFlag('__seed__', seed);
-        core.setFlag('__rand__', seed);
-    }
-    else core.utils.__init_seed();
-    core.clearStatusBar();
+  if (seed != null && seed > 0) {
+    core.setFlag("__seed__", seed);
+    core.setFlag("__rand__", seed);
+  } else core.utils.__init_seed();
+  core.clearStatusBar();
 
-    var todo = [];
-    if (core.flags.startUsingCanvas) {
-        core.hideStatusBar();
-        core.dom.musicBtn.style.display = 'block';
-        core.push(todo, core.firstData.startCanvas);
-    }
-    core.push(todo, { "type": "function", "function": "function() { core.events._startGame_setHard(); }" })
-    core.push(todo, core.firstData.startText);
-    this.insertAction(todo, null, null, function () {
-        core.events._startGame_afterStart(callback);
-    });
+  var todo = [];
+  if (core.flags.startUsingCanvas) {
+    core.hideStatusBar();
+    core.dom.musicBtn.style.display = "block";
+    core.push(todo, core.firstData.startCanvas);
+  }
+  core.push(todo, {
+    type: "function",
+    function: "function() { core.events._startGame_setHard(); }",
+  });
+  core.push(todo, core.firstData.startText);
+  this.insertAction(todo, null, null, function () {
+    core.events._startGame_afterStart(callback);
+  });
 
-    if (route != null) core.startReplay(route);
-}
+  if (route != null) core.startReplay(route);
+};
 
 events.prototype._startGame_setHard = function () {
-    // 根据难度设置flag:hard
-    // 这一段应当在startCanvas之后,startText之前做
-    var hardValue = 0;
-    var hardColor = 'red';
-    main.levelChoose.forEach(function (one) {
-        if (one.name == core.status.hard) {
-            hardValue = one.hard;
-            hardColor = core.arrayToRGBA(one.color || [255, 0, 0, 1]);
-            core.insertAction(one.action);
-        }
-    });
-    core.setFlag('hard', hardValue || 0);
-    core.setFlag('__hardColor__', hardColor);
-}
+  // 根据难度设置flag:hard
+  // 这一段应当在startCanvas之后,startText之前做
+  var hardValue = 0;
+  var hardColor = "red";
+  main.levelChoose.forEach(function (one) {
+    if (one.name == core.status.hard) {
+      hardValue = one.hard;
+      hardColor = core.arrayToRGBA(one.color || [255, 0, 0, 1]);
+      core.insertAction(one.action);
+    }
+  });
+  core.setFlag("hard", hardValue || 0);
+  core.setFlag("__hardColor__", hardColor);
+};
 
 events.prototype._startGame_afterStart = function (callback) {
-    core.ui.closePanel();
-    core.changeFloor(core.firstData.floorId, null, core.firstData.hero.loc, null, function () {
-        // 插入一个空事件避免直接回放录像出错
-        core.insertAction([]);
-        if (callback) callback();
-    });
-    this._startGame_upload();
-}
+  core.ui.closePanel();
+  core.changeFloor(
+    core.firstData.floorId,
+    null,
+    core.firstData.hero.loc,
+    null,
+    function () {
+      // 插入一个空事件避免直接回放录像出错
+      core.insertAction([]);
+      if (callback) callback();
+    }
+  );
+  this._startGame_upload();
+};
 
 events.prototype._startGame_upload = function () {
-    // Upload
-    var formData = new FormData();
-    formData.append('type', 'people');
-    formData.append('name', core.firstData.name);
-    formData.append('version', core.firstData.version);
-    formData.append('platform', core.platform.string);
-    formData.append('hard', core.encodeBase64(core.status.hard));
-    formData.append('hardCode', core.getFlag('hard', 0));
-    formData.append('base64', 1);
+  // Upload
+  var formData = new FormData();
+  formData.append("type", "people");
+  formData.append("name", core.firstData.name);
+  formData.append("version", core.firstData.version);
+  formData.append("platform", core.platform.string);
+  formData.append("hard", core.encodeBase64(core.status.hard));
+  formData.append("hardCode", core.getFlag("hard", 0));
+  formData.append("base64", 1);
 
-    core.utils.http("POST", "/games/upload.php", formData);
-}
+  core.utils.http("POST", "/games/upload.php", formData);
+};
 
 ////// 游戏获胜事件 //////
 events.prototype.win = function (reason, norank, noexit) {
-    if (!noexit) core.status.gameOver = true;
-    return this.eventdata.win(reason, norank, noexit);
-}
+  if (!noexit) core.status.gameOver = true;
+  return this.eventdata.win(reason, norank, noexit);
+};
 
 ////// 游戏失败事件 //////
 events.prototype.lose = function (reason) {
-    if (core.isReplaying()) return core.control._replay_error(reason, function () { core.lose(reason); });
-    core.status.gameOver = true;
-    return this.eventdata.lose(reason);
-}
+  if (core.isReplaying())
+    return core.control._replay_error(reason, function () {
+      core.lose(reason);
+    });
+  core.status.gameOver = true;
+  return this.eventdata.lose(reason);
+};
 
 ////// 游戏结束 //////
 events.prototype.gameOver = function (ending, fromReplay, norank) {
-    if (!core.status.extraEvent) {
-        core.clearMap('all');
-        core.deleteAllCanvas();
-        core.dom.gif2.innerHTML = "";
-        core.setWeather();
-    }
-    core.ui.closePanel();
+  if (!core.status.extraEvent) {
+    core.clearMap("all");
+    core.deleteAllCanvas();
+    core.dom.gif2.innerHTML = "";
+    core.setWeather();
+  }
+  core.ui.closePanel();
 
-    if (main.isCompetition && ending != null) {
-        if (ending == "") ending = "恭喜通关";
-        ending += "[比赛]";
-    }
+  if (main.isCompetition && ending != null) {
+    if (ending == "") ending = "恭喜通关";
+    ending += "[比赛]";
+  }
 
-    var reason = null;
-    if (fromReplay) reason = "录像回放完毕!";
-    else if (core.hasFlag("debug")) reason = "\t[系统提示]调试模式下无法上传成绩";
+  var reason = null;
+  if (fromReplay) reason = "录像回放完毕!";
+  else if (core.hasFlag("debug")) reason = "\t[系统提示]调试模式下无法上传成绩";
 
-    if (reason != null)
-        core.drawText(reason, core.restart);
-    else
-        this._gameOver_confirmUpload(ending, norank);
-}
+  if (reason != null) core.drawText(reason, core.restart);
+  else this._gameOver_confirmUpload(ending, norank);
+};
 
 events.prototype._gameOver_confirmUpload = function (ending, norank) {
-    core.ui.closePanel();
+  core.ui.closePanel();
 
-    if (ending == null) {
-        this._gameOver_confirmDownload(ending);
-        return;
+  if (ending == null) {
+    this._gameOver_confirmDownload(ending);
+    return;
+  }
+  core.ui.drawConfirmBox(
+    "你想记录你的ID和成绩吗?",
+    function () {
+      if (main.isCompetition) {
+        core.events._gameOver_doUpload("", ending, norank);
+      } else {
+        var id = core.getCookie("id") || "";
+        var hint =
+          "请输入你的ID:\n(登录状态下输入数字用户编号可成为蓝名成绩并计入用户通关数)";
+        if (id)
+          hint =
+            "请输入你的ID:\n(输入数字用户编号" +
+            id +
+            "可成为蓝名成绩并计入用户通关数)";
+        core.myprompt(hint, id, function (username) {
+          core.events._gameOver_doUpload(username, ending, norank);
+        });
+      }
+    },
+    function () {
+      if (main.isCompetition) core.events._gameOver_confirmDownload(ending);
+      else core.events._gameOver_doUpload(null, ending, norank);
     }
-    core.ui.drawConfirmBox("你想记录你的ID和成绩吗?", function () {
-        if (main.isCompetition) {
-            core.events._gameOver_doUpload("", ending, norank);
-        }
-        else {
-            var id = core.getCookie('id') || "";
-            var hint = "请输入你的ID:\n(登录状态下输入数字用户编号可成为蓝名成绩并计入用户通关数)";
-            if (id) hint = "请输入你的ID:\n(输入数字用户编号" + id + "可成为蓝名成绩并计入用户通关数)";
-            core.myprompt(hint, id, function (username) {
-                core.events._gameOver_doUpload(username, ending, norank);
-            });
-        }
-    }, function () {
-        if (main.isCompetition)
-            core.events._gameOver_confirmDownload(ending);
-        else
-            core.events._gameOver_doUpload(null, ending, norank);
-    })
-}
+  );
+};
 
 events.prototype._gameOver_doUpload = function (username, ending, norank) {
-    var hp = core.status.hero.hp;
-    if (username == null) hp = 1;
-    core.ui.closePanel();
-    // upload
-    var formData = new FormData();
-    formData.append('type', 'score');
-    formData.append('name', core.firstData.name);
-    formData.append('version', core.firstData.version);
-    formData.append('platform', core.platform.string);
-    formData.append('hard', core.encodeBase64(core.status.hard));
-    formData.append('username', core.encodeBase64(username || ""));
-    formData.append('ending', core.encodeBase64(ending));
-    formData.append('lv', core.status.hero.lv);
-    formData.append('hp', Math.min(hp, Math.pow(2, 63)));
-    formData.append('atk', core.status.hero.atk);
-    formData.append('def', core.status.hero.def);
-    formData.append('mdef', core.status.hero.mdef);
-    formData.append('money', core.status.hero.money);
-    formData.append('experience', core.status.hero.exp);
-    formData.append('steps', core.status.hero.steps);
-    formData.append('norank', norank ? 1 : 0);
-    formData.append('seed', core.getFlag('__seed__'));
-    formData.append('totalTime', Math.floor(core.status.hero.statistics.totalTime / 1000));
-    formData.append('route', core.encodeRoute(core.status.route));
-    formData.append('base64', 1);
+  var hp = core.status.hero.hp;
+  if (username == null) hp = 1;
+  core.ui.closePanel();
+  // upload
+  var formData = new FormData();
+  formData.append("type", "score");
+  formData.append("name", core.firstData.name);
+  formData.append("version", core.firstData.version);
+  formData.append("platform", core.platform.string);
+  formData.append("hard", core.encodeBase64(core.status.hard));
+  formData.append("username", core.encodeBase64(username || ""));
+  formData.append("ending", core.encodeBase64(ending));
+  formData.append("lv", core.status.hero.lv);
+  formData.append("hp", Math.min(hp, Math.pow(2, 63)));
+  formData.append("atk", core.status.hero.atk);
+  formData.append("def", core.status.hero.def);
+  formData.append("mdef", core.status.hero.mdef);
+  formData.append("money", core.status.hero.money);
+  formData.append("experience", core.status.hero.exp);
+  formData.append("steps", core.status.hero.steps);
+  formData.append("norank", norank ? 1 : 0);
+  formData.append("seed", core.getFlag("__seed__"));
+  formData.append(
+    "totalTime",
+    Math.floor(core.status.hero.statistics.totalTime / 1000)
+  );
+  formData.append("route", core.encodeRoute(core.status.route));
+  formData.append("base64", 1);
 
-    if (main.isCompetition)
-        core.http("POST", "/games/competition/upload.php", formData);
-    else
-        core.http("POST", "/games/upload.php", formData);
+  if (main.isCompetition)
+    core.http("POST", "/games/competition/upload.php", formData);
+  else core.http("POST", "/games/upload.php", formData);
 
-    core.events._gameOver_confirmDownload(ending);
-}
+  core.events._gameOver_confirmDownload(ending);
+};
 
 events.prototype._gameOver_confirmDownload = function (ending) {
-    core.ui.closePanel();
-    core.ui.drawConfirmBox("你想下载录像吗?", function () {
-        var obj = {
-            'name': core.firstData.name,
-            'version': core.firstData.version,
-            'hard': core.status.hard,
-            'seed': core.getFlag('__seed__'),
-            'route': core.encodeRoute(core.status.route)
-        }
-        core.download(core.firstData.name + "_" + core.formatDate2(new Date()) + ".h5route",
-            LZString.compressToBase64(JSON.stringify(obj)));
-        core.events._gameOver_askRate(ending);
-    }, function () {
-        core.events._gameOver_askRate(ending);
-    });
-}
+  core.ui.closePanel();
+  core.ui.drawConfirmBox(
+    "你想下载录像吗?",
+    function () {
+      var obj = {
+        name: core.firstData.name,
+        version: core.firstData.version,
+        hard: core.status.hard,
+        seed: core.getFlag("__seed__"),
+        route: core.encodeRoute(core.status.route),
+      };
+      core.download(
+        core.firstData.name + "_" + core.formatDate2(new Date()) + ".h5route",
+        LZString.compressToBase64(JSON.stringify(obj))
+      );
+      core.events._gameOver_askRate(ending);
+    },
+    function () {
+      core.events._gameOver_askRate(ending);
+    }
+  );
+};
 
 events.prototype._gameOver_askRate = function (ending) {
-    core.ui.closePanel();
+  core.ui.closePanel();
 
-    // 继续接下来的事件
-    if (core.status.extraEvent) {
-        core.status.event = core.status.extraEvent;
-        delete core.status.extraEvent;
-        core.lockControl();
-        core.doAction();
-        return;
+  // 继续接下来的事件
+  if (core.status.extraEvent) {
+    core.status.event = core.status.extraEvent;
+    delete core.status.extraEvent;
+    core.lockControl();
+    core.doAction();
+    return;
+  }
+
+  if (ending == null) {
+    if (!core.hasSave(0)) {
+      core.ui.closePanel();
+      core.restart();
+      return;
     }
 
-    if (ending == null) {
-        if (!core.hasSave(0)) {
-            core.ui.closePanel();
-            core.restart();
-            return;
-        }
-
-        core.status.event.selection = 0;
-        core.ui.drawConfirmBox("你想读取自动存档么?", function () {
-            core.ui.closePanel();
-            core.doSL("autoSave", "load");
-        }, function () {
-            core.ui.closePanel();
-            core.restart();
-        });
-        return;
-    }
-
-    core.ui.drawConfirmBox("恭喜通关!你想查看榜单、评论,\n以及评分和标色投票吗?", function () {
-        if (core.platform.isPC) {
-            window.open("/tower/?name=" + core.firstData.name, "_blank");
-            core.restart();
-        }
-        else {
-            window.location.href = "/tower/?name=" + core.firstData.name;
-        }
-    }, function () {
+    core.status.event.selection = 0;
+    core.ui.drawConfirmBox(
+      "你想读取自动存档么?",
+      function () {
+        core.ui.closePanel();
+        core.doSL("autoSave", "load");
+      },
+      function () {
+        core.ui.closePanel();
         core.restart();
-    });
-}
+      }
+    );
+    return;
+  }
+
+  core.ui.drawConfirmBox(
+    "恭喜通关!你想查看榜单、评论,\n以及评分和标色投票吗?",
+    function () {
+      if (core.platform.isPC) {
+        window.open("/tower/?name=" + core.firstData.name, "_blank");
+        core.restart();
+      } else {
+        window.location.href = "/tower/?name=" + core.firstData.name;
+      }
+    },
+    function () {
+      core.restart();
+    }
+  );
+};
 
 ////// 重新开始游戏;此函数将回到标题页面 //////
 events.prototype.restart = function () {
-    core.showStartAnimate();
-    core.playBgm(main.startBgm);
-}
+  core.showStartAnimate();
+  core.playBgm(main.startBgm);
+};
 
 ////// 询问是否需要重新开始 //////
 events.prototype.confirmRestart = function () {
-    core.playSound('打开界面');
-    core.status.event.selection = 1;
-    core.ui.drawConfirmBox("你确定要返回标题页面吗?", function () {
-        core.playSound('确定');
-        core.ui.closePanel();
-        core.restart();
-    }, function () {
-        core.playSound('取消');
-        core.ui.closePanel();
-    });
-}
+  core.playSound("打开界面");
+  core.status.event.selection = 1;
+  core.ui.drawConfirmBox(
+    "你确定要返回标题页面吗?",
+    function () {
+      core.playSound("确定");
+      core.ui.closePanel();
+      core.restart();
+    },
+    function () {
+      core.playSound("取消");
+      core.ui.closePanel();
+    }
+  );
+};
 
 // ------ 系统事件的处理 ------ //
 
 ////// 注册一个系统事件 //////
 // type为事件名,func为事件的处理函数,可接受(data, callback)参数
 events.prototype.registerSystemEvent = function (type, func) {
-    this.systemEvents[type] = func;
-}
+  this.systemEvents[type] = func;
+};
 
 ////// 注销一个系统事件 //////
 events.prototype.unregisterSystemEvent = function (type) {
-    delete this.systemEvents[type];
-}
+  delete this.systemEvents[type];
+};
 
 ////// 执行一个系统事件 //////
 events.prototype.doSystemEvent = function (type, data, callback) {
-    core.clearRouteFolding();
-    if (this.systemEvents[type]) {
-        try {
-            return core.doFunc(this.systemEvents[type], this, data, callback);
-        }
-        catch (e) {
-            console.error(e);
-            console.error("ERROR in systemEvents[" + type + "]");
-        }
+  core.clearRouteFolding();
+  if (this.systemEvents[type]) {
+    try {
+      return core.doFunc(this.systemEvents[type], this, data, callback);
+    } catch (e) {
+      console.error(e);
+      console.error("ERROR in systemEvents[" + type + "]");
     }
-    if (this["_sys_" + type]) return this["_sys_" + type](data, callback);
-    console.error("未知的系统事件: " + type + "!");
-    if (callback) callback();
-}
+  }
+  if (this["_sys_" + type]) return this["_sys_" + type](data, callback);
+  console.error("未知的系统事件: " + type + "!");
+  if (callback) callback();
+};
 
 ////// 触发(x,y)点的事件 //////
 events.prototype.trigger = function (x, y, callback) {
-    var _executeCallback = function () {
-        // 因为trigger之后还有可能触发其他同步脚本(比如阻激夹域检测)
-        // 所以这里强制callback被异步触发
-        if (callback) {
-            setTimeout(callback, 1); // +1是为了录像检测系统 
-        }
-        return;
-    }
-    if (core.status.gameOver) return _executeCallback();
-    if (core.status.event.id == 'action') {
-        core.insertAction({ "type": "function", "function": "function () { core.events._trigger_inAction(" + x + "," + y + "); }", "async": true },
-            null, null, null, true);
-        return _executeCallback();
-    }
-    if (core.status.event.id) return _executeCallback();
-
-    var block = core.getBlock(x, y);
-    if (block == null) return _executeCallback();
-
-    // 执行该点的脚本
-    if (block.event.script) {
-        core.clearRouteFolding();
-        try {
-            eval(block.event.script);
-        } catch (ee) { console.error(ee) }
-    }
-
-    // 碰触事件
-    if (block.event.event) {
-        core.clearRouteFolding();
-        core.insertAction(block.event.event, block.x, block.y);
-        // 不再执行该点的系统事件
-        return _executeCallback();
-    }
-
-    if (block.event.trigger && block.event.trigger != 'null') {
-        var noPass = block.event.noPass, trigger = block.event.trigger;
-        if (noPass) core.clearAutomaticRouteNode(x, y);
-
-        // 转换楼层能否穿透
-        if (trigger == 'changeFloor' && !noPass && this._trigger_ignoreChangeFloor(block))
-            return _executeCallback();
-        core.status.automaticRoute.moveDirectly = false;
-        this.doSystemEvent(trigger, block);
+  var _executeCallback = function () {
+    // 因为trigger之后还有可能触发其他同步脚本(比如阻激夹域检测)
+    // 所以这里强制callback被异步触发
+    if (callback) {
+      setTimeout(callback, 1); // +1是为了录像检测系统
     }
+    return;
+  };
+  if (core.status.gameOver) return _executeCallback();
+  if (core.status.event.id == "action") {
+    core.insertAction(
+      {
+        type: "function",
+        function:
+          "function () { core.events._trigger_inAction(" + x + "," + y + "); }",
+        async: true,
+      },
+      null,
+      null,
+      null,
+      true
+    );
     return _executeCallback();
-}
+  }
+  if (core.status.event.id) return _executeCallback();
+
+  var block = core.getBlock(x, y);
+  if (block == null) return _executeCallback();
+
+  // 执行该点的脚本
+  if (block.event.script) {
+    core.clearRouteFolding();
+    try {
+      eval(block.event.script);
+    } catch (ee) {
+      console.error(ee);
+    }
+  }
+
+  // 碰触事件
+  if (block.event.event) {
+    core.clearRouteFolding();
+    core.insertAction(block.event.event, block.x, block.y);
+    // 不再执行该点的系统事件
+    return _executeCallback();
+  }
+
+  if (block.event.trigger && block.event.trigger != "null") {
+    var noPass = block.event.noPass,
+      trigger = block.event.trigger;
+    if (noPass) core.clearAutomaticRouteNode(x, y);
+
+    // 转换楼层能否穿透
+    if (
+      trigger == "changeFloor" &&
+      !noPass &&
+      this._trigger_ignoreChangeFloor(block)
+    )
+      return _executeCallback();
+    core.status.automaticRoute.moveDirectly = false;
+    this.doSystemEvent(trigger, block);
+  }
+  return _executeCallback();
+};
 
 events.prototype._trigger_inAction = function (x, y) {
-    if (core.status.gameOver || core.status.event.id != 'action') return;
+  if (core.status.gameOver || core.status.event.id != "action") return;
 
-    var block = core.getBlock(x, y);
-    if (block == null) return core.doAction();
+  var block = core.getBlock(x, y);
+  if (block == null) return core.doAction();
 
-    // 执行该点的脚本
-    try {
-        eval(block.event.script);
-    } catch (ee) { console.error(ee) }
+  // 执行该点的脚本
+  try {
+    eval(block.event.script);
+  } catch (ee) {
+    console.error(ee);
+  }
 
-    // 碰触事件
-    if (block.event.event) {
-        core.clearRouteFolding();
-        core.insertAction(block.event.event, block.x, block.y);
-        // 不再执行该点的系统事件
-        return core.doAction();
-    }
-
-    if (block.event.trigger && block.event.trigger != 'null') {
-        this.setEvents(null, x, y);
-        if (block.event.trigger == 'action') {
-            this.insertAction(block.event.data);
-        }
-        else {
-            this.doSystemEvent(block.event.trigger, block, core.doAction);
-            return;
-        }
-    }
+  // 碰触事件
+  if (block.event.event) {
+    core.clearRouteFolding();
+    core.insertAction(block.event.event, block.x, block.y);
+    // 不再执行该点的系统事件
     return core.doAction();
-}
+  }
+
+  if (block.event.trigger && block.event.trigger != "null") {
+    this.setEvents(null, x, y);
+    if (block.event.trigger == "action") {
+      this.insertAction(block.event.data);
+    } else {
+      this.doSystemEvent(block.event.trigger, block, core.doAction);
+      return;
+    }
+  }
+  return core.doAction();
+};
 
 events.prototype._trigger_ignoreChangeFloor = function (block) {
-    var able = core.flags.ignoreChangeFloor;
-    if (block.event.data && block.event.data.ignoreChangeFloor != null)
-        able = block.event.data.ignoreChangeFloor;
-    if (able) {
-        if (core.isReplaying()) {
-            if (core.status.replay.toReplay[0] == 'no') {
-                core.status.replay.toReplay.shift();
-                core.status.route.push("no");
-                return true;
-            }
-        }
-        else if (core.status.automaticRoute.autoHeroMove
-            || core.status.automaticRoute.autoStep < core.status.automaticRoute.autoStepRoutes.length) {
-            core.status.route.push("no");
-            return true;
-        }
+  var able = core.flags.ignoreChangeFloor;
+  if (block.event.data && block.event.data.ignoreChangeFloor != null)
+    able = block.event.data.ignoreChangeFloor;
+  if (able) {
+    if (core.isReplaying()) {
+      if (core.status.replay.toReplay[0] == "no") {
+        core.status.replay.toReplay.shift();
+        core.status.route.push("no");
+        return true;
+      }
+    } else if (
+      core.status.automaticRoute.autoHeroMove ||
+      core.status.automaticRoute.autoStep <
+        core.status.automaticRoute.autoStepRoutes.length
+    ) {
+      core.status.route.push("no");
+      return true;
     }
-    return false;
-}
+  }
+  return false;
+};
 
 events.prototype._sys_battle = function (data, callback) {
-    // 检查是否需要改变朝向
-    /* if (data.x == core.nextX() && data.y == core.nextY()) {
+  // 检查是否需要改变朝向
+  /* if (data.x == core.nextX() && data.y == core.nextY()) {
         var dir = core.turnDirection(":back");
         var id = data.event.id, toId = (data.event.faceIds || {})[dir];
         if (toId && id != toId) {
@@ -433,499 +489,554 @@ events.prototype._sys_battle = function (data, callback) {
         }
     } */
 
-    // 检查战前事件
-    var beforeBattle = [];
-    core.push(beforeBattle, core.floors[core.status.floorId].beforeBattle[data.x + "," + data.y]);
-    core.push(beforeBattle, (core.material.enemys[data.event.id] || {}).beforeBattle);
-    if (beforeBattle.length > 0) {
-        core.push(beforeBattle, [{ "type": "battle", "x": data.x, "y": data.y }]);
-        core.clearContinueAutomaticRoute();
+  // 检查战前事件
+  var beforeBattle = [];
+  core.push(
+    beforeBattle,
+    core.floors[core.status.floorId].beforeBattle[data.x + "," + data.y]
+  );
+  core.push(
+    beforeBattle,
+    (core.material.enemys[data.event.id] || {}).beforeBattle
+  );
+  if (beforeBattle.length > 0) {
+    core.push(beforeBattle, [{ type: "battle", x: data.x, y: data.y }]);
+    core.clearContinueAutomaticRoute();
 
-        // 自动存档
-        var inAction = core.status.event.id == 'action';
-        if (inAction) {
-            core.insertAction(beforeBattle, data.x, data.y);
-            core.doAction();
-        } else {
-            core.autosave(true);
-            core.insertAction(beforeBattle, data.x, data.y, callback);
-        }
+    // 自动存档
+    var inAction = core.status.event.id == "action";
+    if (inAction) {
+      core.insertAction(beforeBattle, data.x, data.y);
+      core.doAction();
     } else {
-        this.battle(data.event.id, data.x, data.y, false, callback);
+      core.autosave(true);
+      core.insertAction(beforeBattle, data.x, data.y, callback);
     }
-}
+  } else {
+    this.battle(data.event.id, data.x, data.y, false, callback);
+  }
+};
 
 ////// 战斗 //////
 events.prototype.battle = function (id, x, y, force, callback) {
-    core.saveAndStopAutomaticRoute();
-    id = id || core.getBlockId(x, y);
-    if (!id) return core.clearContinueAutomaticRoute(callback);
-    // 非强制战斗
-    if (!core.enemys.canBattle(id, x, y) && !force && !core.status.event.id) {
-        core.stopSound();
-        core.playSound('操作失败');
-        core.drawTip("你打不过此怪物!", id);
-        return core.clearContinueAutomaticRoute(callback);
-    }
-    // 自动存档
-    if (!core.status.event.id) core.autosave(true);
-    // 战前事件
-    if (!this.beforeBattle(id, x, y))
-        return core.clearContinueAutomaticRoute(callback);
-    // 战后事件
-    this.afterBattle(id, x, y);
-    if (callback) callback();
-}
+  core.saveAndStopAutomaticRoute();
+  id = id || core.getBlockId(x, y);
+  if (!id) return core.clearContinueAutomaticRoute(callback);
+  // 非强制战斗
+  if (!core.enemys.canBattle(id, x, y) && !force && !core.status.event.id) {
+    core.stopSound();
+    core.playSound("操作失败");
+    core.drawTip("你打不过此怪物!", id);
+    return core.clearContinueAutomaticRoute(callback);
+  }
+  // 自动存档
+  if (!core.status.event.id) core.autosave(true);
+  // 战前事件
+  if (!this.beforeBattle(id, x, y))
+    return core.clearContinueAutomaticRoute(callback);
+  // 战后事件
+  this.afterBattle(id, x, y);
+  if (callback) callback();
+};
 
 ////// 战斗前触发的事件 //////
 events.prototype.beforeBattle = function (enemyId, x, y) {
-    return this.eventdata.beforeBattle(enemyId, x, y)
-}
+  return this.eventdata.beforeBattle(enemyId, x, y);
+};
 
 ////// 战斗结束后触发的事件 //////
 events.prototype.afterBattle = function (enemyId, x, y) {
-    return this.eventdata.afterBattle(enemyId, x, y);
-}
+  return this.eventdata.afterBattle(enemyId, x, y);
+};
 
 events.prototype._sys_openDoor = function (data, callback) {
-    this.openDoor(data.x, data.y, true, function () {
-        core.replay();
-        if (callback) callback();
-    });
-}
+  this.openDoor(data.x, data.y, true, function () {
+    core.replay();
+    if (callback) callback();
+  });
+};
 
 ////// 开门 //////
 events.prototype.openDoor = function (x, y, needKey, callback) {
-    var block = core.getBlock(x, y);
-    core.saveAndStopAutomaticRoute();
-    if (!this._openDoor_check(block, x, y, needKey)) {
-        var locked = core.status.lockControl;
-        core.waitHeroToStop(function () {
-            if (!locked) core.unlockControl();
-            if (callback) callback();
-        });
-        return;
-    }
-    if (core.status.replay.speed == 24) {
-        core.status.replay.animate = true;
-        core.removeBlock(x, y);
-        setTimeout(function () {
-            core.status.replay.animate = false;
-            core.events.afterOpenDoor(block.event.id, x, y);
-            if (callback) callback();
-        }, 1); // +1是为了录像检测系统
-    } else {
-        this._openDoor_animate(block, x, y, callback);
-    }
-}
-
-events.prototype._openDoor_check = function (block, x, y, needKey) {
-    var clearAndReturn = function () {
-        core.clearContinueAutomaticRoute();
-        return false;
-    }
-
-    if (block == null || block.event == null) return clearAndReturn();
-    var id = block.event.id;
-
-    // 是否存在门或暗墙
-    if (core.material.icons.animates[id] == null && core.material.icons.npc48[id] == null) {
-        return clearAndReturn();
-    }
-
-    if (id == 'steelDoor' && core.flags.steelDoorWithoutKey)
-        needKey = false;
-    var doorInfo = block.event.doorInfo;
-    if (doorInfo == null) return clearAndReturn();
-    // Check all keys
-    var keyInfo = doorInfo.keys || {};
-    if (needKey) {
-        for (var keyName in keyInfo) {
-            var keyValue = keyInfo[keyName];
-            if (keyName.endsWith(':o')) keyName = keyName.substring(0, keyName.length - 2);
-
-            // --- 如果是一个不存在的道具,则直接认为无法开启
-            if (!core.material.items[keyName]) {
-                core.stopSound();
-                core.playSound('操作失败');
-                core.drawTip("无法开启此门");
-                return clearAndReturn();
-            }
-            if (core.itemCount(keyName) < keyValue) {
-                core.stopSound();
-                core.playSound('操作失败');
-                core.drawTip("你的" + ((core.material.items[keyName] || {}).name || "钥匙") + "不足!", null, true);
-                return false;
-            }
-        }
-        if (!core.status.event.id) core.autosave(true);
-        for (var keyName in keyInfo) {
-            if (!keyName.endsWith(':o')) core.removeItem(keyName, keyInfo[keyName]);
-        }
-    }
-    core.playSound(doorInfo.openSound);
-    return true;
-}
-
-events.prototype._openDoor_animate = function (block, x, y, callback) {
-    var blockInfo = core.getBlockInfo(block);
-    blockInfo.opacity = block.opacity;
-    blockInfo.filter = block.filter;
-
-    var speed = (block.event.doorInfo.time || 160) / 4;
-
+  var block = core.getBlock(x, y);
+  core.saveAndStopAutomaticRoute();
+  if (!this._openDoor_check(block, x, y, needKey)) {
     var locked = core.status.lockControl;
-    core.lockControl();
+    core.waitHeroToStop(function () {
+      if (!locked) core.unlockControl();
+      if (callback) callback();
+    });
+    return;
+  }
+  if (core.status.replay.speed == 24) {
     core.status.replay.animate = true;
     core.removeBlock(x, y);
+    setTimeout(function () {
+      core.status.replay.animate = false;
+      core.events.afterOpenDoor(block.event.id, x, y);
+      if (callback) callback();
+    }, 1); // +1是为了录像检测系统
+  } else {
+    this._openDoor_animate(block, x, y, callback);
+  }
+};
 
-    blockInfo.posX = 0;
-    core.maps._drawBlockInfo(blockInfo, x, y);
+events.prototype._openDoor_check = function (block, x, y, needKey) {
+  var clearAndReturn = function () {
+    core.clearContinueAutomaticRoute();
+    return false;
+  };
 
-    var cb = function () {
-        core.maps._removeBlockFromMap(core.status.floorId, block);
-        if (!locked) core.unlockControl();
-        core.status.replay.animate = false;
-        core.events.afterOpenDoor(block.event.id, x, y);
-        if (callback) callback();
+  if (block == null || block.event == null) return clearAndReturn();
+  var id = block.event.id;
+
+  // 是否存在门或暗墙
+  if (
+    core.material.icons.animates[id] == null &&
+    core.material.icons.npc48[id] == null
+  ) {
+    return clearAndReturn();
+  }
+
+  if (id == "steelDoor" && core.flags.steelDoorWithoutKey) needKey = false;
+  var doorInfo = block.event.doorInfo;
+  if (doorInfo == null) return clearAndReturn();
+  // Check all keys
+  var keyInfo = doorInfo.keys || {};
+  if (needKey) {
+    for (var keyName in keyInfo) {
+      var keyValue = keyInfo[keyName];
+      if (keyName.endsWith(":o"))
+        keyName = keyName.substring(0, keyName.length - 2);
+
+      // --- 如果是一个不存在的道具,则直接认为无法开启
+      if (!core.material.items[keyName]) {
+        core.stopSound();
+        core.playSound("操作失败");
+        core.drawTip("无法开启此门");
+        return clearAndReturn();
+      }
+      if (core.itemCount(keyName) < keyValue) {
+        core.stopSound();
+        core.playSound("操作失败");
+        core.drawTip(
+          "你的" +
+            ((core.material.items[keyName] || {}).name || "钥匙") +
+            "不足!",
+          null,
+          true
+        );
+        return false;
+      }
     }
+    if (!core.status.event.id) core.autosave(true);
+    for (var keyName in keyInfo) {
+      if (!keyName.endsWith(":o")) core.removeItem(keyName, keyInfo[keyName]);
+    }
+  }
+  core.playSound(doorInfo.openSound);
+  return true;
+};
 
-    var animate = window.setInterval(function () {
-        blockInfo.posX++;
-        if (blockInfo.posX == 4) {
-            clearInterval(animate);
-            delete core.animateFrame.asyncId[animate];
-            cb();
-            return;
-        }
-        core.maps._drawBlockInfo(blockInfo, x, y);
-    }, core.status.replay.speed == 24 ? 1 : speed / Math.max(core.status.replay.speed, 1));
+events.prototype._openDoor_animate = function (block, x, y, callback) {
+  var blockInfo = core.getBlockInfo(block);
+  blockInfo.opacity = block.opacity;
+  blockInfo.filter = block.filter;
 
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = cb;
-}
+  var speed = (block.event.doorInfo.time || 160) / 4;
+
+  var locked = core.status.lockControl;
+  core.lockControl();
+  core.status.replay.animate = true;
+  core.removeBlock(x, y);
+
+  blockInfo.posX = 0;
+  core.maps._drawBlockInfo(blockInfo, x, y);
+
+  var cb = function () {
+    core.maps._removeBlockFromMap(core.status.floorId, block);
+    if (!locked) core.unlockControl();
+    core.status.replay.animate = false;
+    core.events.afterOpenDoor(block.event.id, x, y);
+    if (callback) callback();
+  };
+
+  var animate = window.setInterval(
+    function () {
+      blockInfo.posX++;
+      if (blockInfo.posX == 4) {
+        clearInterval(animate);
+        delete core.animateFrame.asyncId[animate];
+        cb();
+        return;
+      }
+      core.maps._drawBlockInfo(blockInfo, x, y);
+    },
+    core.status.replay.speed == 24
+      ? 1
+      : speed / Math.max(core.status.replay.speed, 1)
+  );
+
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = cb;
+};
 
 ////// 开一个门后触发的事件 //////
 events.prototype.afterOpenDoor = function (doorId, x, y) {
-    return this.eventdata.afterOpenDoor(doorId, x, y);
-}
+  return this.eventdata.afterOpenDoor(doorId, x, y);
+};
 
 events.prototype._sys_getItem = function (data, callback) {
-    this.getItem(data.event.id, 1, data.x, data.y, false, callback);
-}
+  this.getItem(data.event.id, 1, data.x, data.y, false, callback);
+};
 
 ////// 获得某个物品 //////
 events.prototype.getItem = function (id, num, x, y, isGentleClick, callback) {
-    if (num == null) num = 1;
-    var itemCls = core.material.items[id].cls;
-    core.removeBlock(x, y);
-    core.items.getItemEffect(id, num);
-    var text = '获得 ' + core.material.items[id].name;
-    if (num > 1) text += "x" + num;
-    if (itemCls === 'items' && num == 1) text += core.items.getItemEffectTip(id);
-    core.drawTip(text, id);
+  if (num == null) num = 1;
+  var itemCls = core.material.items[id].cls;
+  core.removeBlock(x, y);
+  core.items.getItemEffect(id, num);
+  var text = "获得 " + core.material.items[id].name;
+  if (num > 1) text += "x" + num;
+  if (itemCls === "items" && num == 1) text += core.items.getItemEffectTip(id);
+  core.drawTip(text, id);
 
-    // --- 首次获得道具的提示
-    if (!core.hasFlag("__itemHint__")) core.setFlag("__itemHint__", []);
-    var itemHint = core.getFlag("__itemHint__");
-    if (core.flags.itemFirstText && itemHint.indexOf(id) < 0 && itemCls != 'items') {
-        var hint = core.material.items[id].text || "该道具暂无描述";
-        try {
-            hint = core.replaceText(hint);
-        } catch (e) { }
-        if (!core.status.event.id || core.status.event.id == 'action') {
-            core.insertAction("\t[" + core.material.items[id].name + "," + id + "]" + hint + "\n"
-                + (id.endsWith('Key') ? "(钥匙类道具,遇到对应的门时自动打开)"
-                    : itemCls == 'tools' ? "(消耗类道具,请按T在道具栏使用)"
-                        : itemCls == 'constants' ? "(永久类道具,请按T在道具栏使用)"
-                            : itemCls == 'equips' ? "(装备类道具,请按Q在装备栏进行装备)" : ""));
-        }
-        itemHint.push(id);
+  // --- 首次获得道具的提示
+  if (!core.hasFlag("__itemHint__")) core.setFlag("__itemHint__", []);
+  var itemHint = core.getFlag("__itemHint__");
+  if (
+    core.flags.itemFirstText &&
+    itemHint.indexOf(id) < 0 &&
+    itemCls != "items"
+  ) {
+    var hint = core.material.items[id].text || "该道具暂无描述";
+    try {
+      hint = core.replaceText(hint);
+    } catch (e) {}
+    if (!core.status.event.id || core.status.event.id == "action") {
+      core.insertAction(
+        "\t[" +
+          core.material.items[id].name +
+          "," +
+          id +
+          "]" +
+          hint +
+          "\n" +
+          (id.endsWith("Key")
+            ? "(钥匙类道具,遇到对应的门时自动打开)"
+            : itemCls == "tools"
+            ? "(消耗类道具,请按T在道具栏使用)"
+            : itemCls == "constants"
+            ? "(永久类道具,请按T在道具栏使用)"
+            : itemCls == "equips"
+            ? "(装备类道具,请按Q在装备栏进行装备)"
+            : "")
+      );
     }
+    itemHint.push(id);
+  }
 
-    this.afterGetItem(id, x, y, isGentleClick);
-    if (callback) callback();
-}
+  this.afterGetItem(id, x, y, isGentleClick);
+  if (callback) callback();
+};
 
 events.prototype.afterGetItem = function (id, x, y, isGentleClick) {
-    this.eventdata.afterGetItem(id, x, y, isGentleClick);
-}
+  this.eventdata.afterGetItem(id, x, y, isGentleClick);
+};
 
 ////// 获得面前的物品(轻按) //////
 events.prototype.getNextItem = function (noRoute) {
-    if (core.isMoving() || !core.flags.enableGentleClick) return false;
-    if (this._canGetNextItem()) return this._getNextItem(null, noRoute);
+  if (core.isMoving() || !core.flags.enableGentleClick) return false;
+  if (this._canGetNextItem()) return this._getNextItem(null, noRoute);
 
-    var directions = ["up", "down", "left", "right"].filter(function (dir) {
-        return core.events._canGetNextItem(dir);
-    });
-    return directions.length > 0 ? this._getNextItem(directions[0], noRoute) : false;
-}
+  var directions = ["up", "down", "left", "right"].filter(function (dir) {
+    return core.events._canGetNextItem(dir);
+  });
+  return directions.length > 0
+    ? this._getNextItem(directions[0], noRoute)
+    : false;
+};
 
 events.prototype._canGetNextItem = function (direction) {
-    direction = direction || core.getHeroLoc('direction');
-    if (!core.canMoveHero(null, null, direction)) return;
-    var nx = core.getHeroLoc('x') + core.utils.scan[direction].x;
-    var ny = core.getHeroLoc('y') + core.utils.scan[direction].y;
-    var block = core.getBlock(nx, ny);
-    return block != null && !block.event.script && !block.event.event && block.event.trigger == 'getItem';
-}
+  direction = direction || core.getHeroLoc("direction");
+  if (!core.canMoveHero(null, null, direction)) return;
+  var nx = core.getHeroLoc("x") + core.utils.scan[direction].x;
+  var ny = core.getHeroLoc("y") + core.utils.scan[direction].y;
+  var block = core.getBlock(nx, ny);
+  return (
+    block != null &&
+    !block.event.script &&
+    !block.event.event &&
+    block.event.trigger == "getItem"
+  );
+};
 
 events.prototype._getNextItem = function (direction, noRoute) {
-    direction = direction || core.getHeroLoc('direction');
-    var nx = core.getHeroLoc('x') + core.utils.scan[direction].x;
-    var ny = core.getHeroLoc('y') + core.utils.scan[direction].y;
-    if (!noRoute) core.status.route.push("getNext");
-    this.getItem(core.getBlockId(nx, ny), 1, nx, ny, true);
-    return true;
-}
+  direction = direction || core.getHeroLoc("direction");
+  var nx = core.getHeroLoc("x") + core.utils.scan[direction].x;
+  var ny = core.getHeroLoc("y") + core.utils.scan[direction].y;
+  if (!noRoute) core.status.route.push("getNext");
+  this.getItem(core.getBlockId(nx, ny), 1, nx, ny, true);
+  return true;
+};
 
 events.prototype._sys_changeFloor = function (data, callback) {
-    data = data.event.data;
-    var heroLoc = {};
-    if (data.loc) heroLoc = { 'x': data.loc[0], 'y': data.loc[1] };
-    if (data.direction) heroLoc.direction = data.direction;
-    if (core.status.event.id != 'action') core.status.event.id = null;
-    core.changeFloor(data.floorId, data.stair, heroLoc, data.time, function () {
-        core.replay();
-        if (callback) callback();
-    });
-}
+  data = data.event.data;
+  var heroLoc = {};
+  if (data.loc) heroLoc = { x: data.loc[0], y: data.loc[1] };
+  if (data.direction) heroLoc.direction = data.direction;
+  if (core.status.event.id != "action") core.status.event.id = null;
+  core.changeFloor(data.floorId, data.stair, heroLoc, data.time, function () {
+    core.replay();
+    if (callback) callback();
+  });
+};
 
 ////// 楼层切换 //////
-events.prototype.changeFloor = function (floorId, stair, heroLoc, time, callback) {
-    var info = this._changeFloor_getInfo(floorId, stair, heroLoc, time);
-    if (info == null) {
-        if (callback) callback();
-        return;
-    }
-    floorId = info.floorId;
-    info.locked = core.status.lockControl;
+events.prototype.changeFloor = function (
+  floorId,
+  stair,
+  heroLoc,
+  time,
+  callback
+) {
+  var info = this._changeFloor_getInfo(floorId, stair, heroLoc, time);
+  if (info == null) {
+    if (callback) callback();
+    return;
+  }
+  floorId = info.floorId;
+  info.locked = core.status.lockControl;
 
-    core.dom.floorNameLabel.innerText = core.status.maps[floorId].title;
-    core.lockControl();
-    core.stopAutomaticRoute();
-    core.clearContinueAutomaticRoute();
-    core.status.replay.animate = true;
-    clearInterval(core.interval.onDownInterval);
-    core.interval.onDownInterval = 'tmp';
+  core.dom.floorNameLabel.innerText = core.status.maps[floorId].title;
+  core.lockControl();
+  core.stopAutomaticRoute();
+  core.clearContinueAutomaticRoute();
+  core.status.replay.animate = true;
+  clearInterval(core.interval.onDownInterval);
+  core.interval.onDownInterval = "tmp";
 
-    this._changeFloor_beforeChange(info, callback);
-}
+  this._changeFloor_beforeChange(info, callback);
+};
 
-events.prototype._changeFloor_getInfo = function (floorId, stair, heroLoc, time) {
-    floorId = floorId || core.status.floorId;
-    if (floorId == ':before') {
-        var index = core.floorIds.indexOf(core.status.floorId);
-        if (index > 0) floorId = core.floorIds[index - 1];
-        else floorId = core.status.floorId;
-    }
-    else if (floorId == ':next') {
-        var index = core.floorIds.indexOf(core.status.floorId);
-        if (index < core.floorIds.length - 1) floorId = core.floorIds[index + 1];
-        else floorId = core.status.floorId;
-    } else if (floorId == ':now') {
-        floorId = core.status.floorId;
-    }
-    if (!core.status.maps[floorId]) {
-        console.error("不存在的楼层:" + floorId);
-        return null;
-    }
+events.prototype._changeFloor_getInfo = function (
+  floorId,
+  stair,
+  heroLoc,
+  time
+) {
+  floorId = floorId || core.status.floorId;
+  if (floorId == ":before") {
+    var index = core.floorIds.indexOf(core.status.floorId);
+    if (index > 0) floorId = core.floorIds[index - 1];
+    else floorId = core.status.floorId;
+  } else if (floorId == ":next") {
+    var index = core.floorIds.indexOf(core.status.floorId);
+    if (index < core.floorIds.length - 1) floorId = core.floorIds[index + 1];
+    else floorId = core.status.floorId;
+  } else if (floorId == ":now") {
+    floorId = core.status.floorId;
+  }
+  if (!core.status.maps[floorId]) {
+    console.error("不存在的楼层:" + floorId);
+    return null;
+  }
 
-    if (main.mode != 'play' || core.isReplaying()) time = 0;
-    if (time == null) time = core.values.floorChangeTime;
-    time /= 20;
+  if (main.mode != "play" || core.isReplaying()) time = 0;
+  if (time == null) time = core.values.floorChangeTime;
+  time /= 20;
 
-    return {
-        floorId: floorId,
-        time: time,
-        heroLoc: core.clone(this._changeFloor_getHeroLoc(floorId, stair, heroLoc))
-    };
-}
+  return {
+    floorId: floorId,
+    time: time,
+    heroLoc: core.clone(this._changeFloor_getHeroLoc(floorId, stair, heroLoc)),
+  };
+};
 
 events.prototype._changeFloor_getHeroLoc = function (floorId, stair, heroLoc) {
-    if (!heroLoc)
-        heroLoc = core.clone(core.status.hero.loc);
-    if (stair) {
-        // --- 对称
-        if (stair == ':now')
-            heroLoc = core.clone(core.status.hero.loc);
-        else if (stair == ':symmetry') {
-            heroLoc.x = core.bigmap.width - 1 - core.getHeroLoc('x');
-            heroLoc.y = core.bigmap.height - 1 - core.getHeroLoc('y');
-        }
-        else if (stair == ':symmetry_x')
-            heroLoc.x = core.bigmap.width - 1 - core.getHeroLoc('x');
-        else if (stair == ':symmetry_y')
-            heroLoc.y = core.bigmap.height - 1 - core.getHeroLoc('y');
-        // 检查该层地图的 upFloor & downFloor & flyPoint
-        else if (core.status.maps[floorId][stair]) {
-            heroLoc.x = core.status.maps[floorId][stair][0];
-            heroLoc.y = core.status.maps[floorId][stair][1];
-        }
-        else {
-            core.extractBlocks(floorId);
-            var blocks = core.status.maps[floorId].blocks;
-            for (var i in blocks) {
-                if (!blocks[i].disable && blocks[i].event.id === stair) {
-                    heroLoc.x = blocks[i].x;
-                    heroLoc.y = blocks[i].y;
-                    break;
-                }
-            }
+  if (!heroLoc) heroLoc = core.clone(core.status.hero.loc);
+  if (stair) {
+    // --- 对称
+    if (stair == ":now") heroLoc = core.clone(core.status.hero.loc);
+    else if (stair == ":symmetry") {
+      heroLoc.x = core.bigmap.width - 1 - core.getHeroLoc("x");
+      heroLoc.y = core.bigmap.height - 1 - core.getHeroLoc("y");
+    } else if (stair == ":symmetry_x")
+      heroLoc.x = core.bigmap.width - 1 - core.getHeroLoc("x");
+    else if (stair == ":symmetry_y")
+      heroLoc.y = core.bigmap.height - 1 - core.getHeroLoc("y");
+    // 检查该层地图的 upFloor & downFloor & flyPoint
+    else if (core.status.maps[floorId][stair]) {
+      heroLoc.x = core.status.maps[floorId][stair][0];
+      heroLoc.y = core.status.maps[floorId][stair][1];
+    } else {
+      core.extractBlocks(floorId);
+      var blocks = core.status.maps[floorId].blocks;
+      for (var i in blocks) {
+        if (!blocks[i].disable && blocks[i].event.id === stair) {
+          heroLoc.x = blocks[i].x;
+          heroLoc.y = blocks[i].y;
+          break;
         }
+      }
     }
-    ['x', 'y', 'direction'].forEach(function (name) {
-        if (heroLoc[name] == null)
-            heroLoc[name] = core.getHeroLoc(name);
-    });
-    return heroLoc;
-}
+  }
+  ["x", "y", "direction"].forEach(function (name) {
+    if (heroLoc[name] == null) heroLoc[name] = core.getHeroLoc(name);
+  });
+  return heroLoc;
+};
 
 events.prototype._changeFloor_beforeChange = function (info, callback) {
-    this._changeFloor_playSound();
-    // 需要 setTimeout 执行,不然会出错
-    window.setTimeout(function () {
-        if (info.time == 0)
-            core.events._changeFloor_changing(info, callback);
-        else
-            core.showWithAnimate(core.dom.floorMsgGroup, info.time / 2, function () {
-                core.events._changeFloor_changing(info, callback);
-            });
-    }, 25)
-}
+  this._changeFloor_playSound();
+  // 需要 setTimeout 执行,不然会出错
+  window.setTimeout(function () {
+    if (info.time == 0) core.events._changeFloor_changing(info, callback);
+    else
+      core.showWithAnimate(core.dom.floorMsgGroup, info.time / 2, function () {
+        core.events._changeFloor_changing(info, callback);
+      });
+  }, 25);
+};
 
 events.prototype._changeFloor_playSound = function () {
-    // 播放换层音效
-    if (core.hasFlag('__fromLoad__')) // 是否是读档造成的切换
-        core.playSound('读档');
-    else if (core.hasFlag('__isFlying__')) // 是否是楼传造成的切换
-        core.playSound('飞行器');
-    else
-        core.playSound('上下楼');
-}
+  // 播放换层音效
+  if (core.hasFlag("__fromLoad__"))
+    // 是否是读档造成的切换
+    core.playSound("读档");
+  else if (core.hasFlag("__isFlying__"))
+    // 是否是楼传造成的切换
+    core.playSound("飞行器");
+  else core.playSound("上下楼");
+};
 
 events.prototype._changeFloor_changing = function (info, callback) {
-    this.changingFloor(info.floorId, info.heroLoc);
-    // 回归视角
-    var __lockViewport__ = flags.__lockViewport__;
-    core.setFlag('__lockViewport__', null);
-    core.drawHero();
-    core.setFlag('__lockViewport__', __lockViewport__);
+  this.changingFloor(info.floorId, info.heroLoc);
+  // 回归视角
+  var __lockViewport__ = flags.__lockViewport__;
+  core.setFlag("__lockViewport__", null);
+  core.drawHero();
+  core.setFlag("__lockViewport__", __lockViewport__);
 
-    if (info.time == 0)
-        this._changeFloor_afterChange(info, callback);
-    else
-        core.hideWithAnimate(core.dom.floorMsgGroup, info.time / 4, function () {
-            core.events._changeFloor_afterChange(info, callback);
-        });
-}
+  if (info.time == 0) this._changeFloor_afterChange(info, callback);
+  else
+    core.hideWithAnimate(core.dom.floorMsgGroup, info.time / 4, function () {
+      core.events._changeFloor_afterChange(info, callback);
+    });
+};
 
 events.prototype._changeFloor_afterChange = function (info, callback) {
-    if (!info.locked) core.unlockControl();
-    core.status.replay.animate = false;
-    core.events.afterChangeFloor(info.floorId);
+  if (!info.locked) core.unlockControl();
+  core.status.replay.animate = false;
+  core.events.afterChangeFloor(info.floorId);
 
-    if (callback) callback();
-}
+  if (callback) callback();
+};
 
 events.prototype.changingFloor = function (floorId, heroLoc) {
-    this.eventdata.changingFloor(floorId, heroLoc);
-}
+  this.eventdata.changingFloor(floorId, heroLoc);
+};
 
 ////// 转换楼层结束的事件 //////
 events.prototype.afterChangeFloor = function (floorId) {
-    if (main.mode != 'play') return;
-    return this.eventdata.afterChangeFloor(floorId);
-}
+  if (main.mode != "play") return;
+  return this.eventdata.afterChangeFloor(floorId);
+};
 
 ////// 是否到达过某个楼层 //////
 events.prototype.hasVisitedFloor = function (floorId) {
-    if (!core.hasFlag("__visited__")) core.setFlag("__visited__", {});
-    return core.getFlag("__visited__")[floorId] || false;
-}
+  if (!core.hasFlag("__visited__")) core.setFlag("__visited__", {});
+  return core.getFlag("__visited__")[floorId] || false;
+};
 
 ////// 到达某楼层 //////
 events.prototype.visitFloor = function (floorId) {
-    if (!core.hasFlag("__visited__")) core.setFlag("__visited__", {});
-    core.getFlag("__visited__")[floorId] = true;
-}
+  if (!core.hasFlag("__visited__")) core.setFlag("__visited__", {});
+  core.getFlag("__visited__")[floorId] = true;
+};
 
 events.prototype._sys_pushBox = function (data, callback) {
-    this.pushBox(data);
-    if (callback) callback();
-}
+  this.pushBox(data);
+  if (callback) callback();
+};
 
 ////// 推箱子 //////
 events.prototype.pushBox = function (data) {
-    if (data.event.id != 'box' && data.event.id != 'boxed') return;
+  if (data.event.id != "box" && data.event.id != "boxed") return;
 
-    // 判断还能否前进,看看是否存在事件
-    var direction = core.getHeroLoc('direction'),
-        nx = data.x + core.utils.scan[direction].x, ny = data.y + core.utils.scan[direction].y;
+  // 判断还能否前进,看看是否存在事件
+  var direction = core.getHeroLoc("direction"),
+    nx = data.x + core.utils.scan[direction].x,
+    ny = data.y + core.utils.scan[direction].y;
 
-    // 检测能否推上去
-    if (!core.canMoveHero()) return;
-    var canGoDeadZone = core.flags.canGoDeadZone;
-    core.flags.canGoDeadZone = true;
-    if (!core.canMoveHero(data.x, data.y, direction)) {
-        core.flags.canGoDeadZone = canGoDeadZone;
-        return;
-    }
+  // 检测能否推上去
+  if (!core.canMoveHero()) return;
+  var canGoDeadZone = core.flags.canGoDeadZone;
+  core.flags.canGoDeadZone = true;
+  if (!core.canMoveHero(data.x, data.y, direction)) {
     core.flags.canGoDeadZone = canGoDeadZone;
+    return;
+  }
+  core.flags.canGoDeadZone = canGoDeadZone;
 
-    var nextId = core.getBlockId(nx, ny);
-    if (nextId != null && nextId != 'flower') return;
+  var nextId = core.getBlockId(nx, ny);
+  if (nextId != null && nextId != "flower") return;
 
-    core.setBlock(nextId == null ? 'box' : 'boxed', nx, ny);
+  core.setBlock(nextId == null ? "box" : "boxed", nx, ny);
 
-    if (data.event.id == 'box')
-        core.removeBlock(data.x, data.y);
-    else
-        core.setBlock('flower', data.x, data.y);
-    // 勇士前进一格,然后触发推箱子后事件
-    core.insertAction([
-        { "type": "moveAction" },
-        { "type": "function", "function": "function() { core.afterPushBox(); }" }
-    ]);
-}
+  if (data.event.id == "box") core.removeBlock(data.x, data.y);
+  else core.setBlock("flower", data.x, data.y);
+  // 勇士前进一格,然后触发推箱子后事件
+  core.insertAction([
+    { type: "moveAction" },
+    { type: "function", function: "function() { core.afterPushBox(); }" },
+  ]);
+};
 
 ////// 推箱子后的事件 //////
 events.prototype.afterPushBox = function () {
-    return this.eventdata.afterPushBox();
-}
+  return this.eventdata.afterPushBox();
+};
 
 events.prototype._sys_ski = function (data, callback) {
-    core.insertAction(["V2.6后,请将滑冰放在背景层!"], data.x, data.y);
-    if (callback) callback();
-}
+  core.insertAction(["V2.6后,请将滑冰放在背景层!"], data.x, data.y);
+  if (callback) callback();
+};
 
 /// 当前是否在冰上
 events.prototype.onSki = function (number) {
-    if (number == null) number = core.getBgNumber();
-    var block = core.getBlockByNumber(number);
-    return block && block.event && block.event.trigger == 'ski';
-}
+  if (number == null) number = core.getBgNumber();
+  var block = core.getBlockByNumber(number);
+  return block && block.event && block.event.trigger == "ski";
+};
 
 events.prototype._sys_action = function (data, callback) {
-    var ev = core.clone(data.event.data), ex = data.x, ey = data.y;
-    // 检查是否需要改变朝向
-    if (ex == core.nextX() && ey == core.nextY()) {
-        var dir = core.turnDirection(":back");
-        var id = data.event.id, toId = (data.event.faceIds || {})[dir];
-        if (toId && id != toId) {
-            var number = core.getNumberById(toId);
-            if (number > 0)
-                core.setBlock(number, ex, ey);
-        }
+  var ev = core.clone(data.event.data),
+    ex = data.x,
+    ey = data.y;
+  // 检查是否需要改变朝向
+  if (ex == core.nextX() && ey == core.nextY()) {
+    var dir = core.turnDirection(":back");
+    var id = data.event.id,
+      toId = (data.event.faceIds || {})[dir];
+    if (toId && id != toId) {
+      var number = core.getNumberById(toId);
+      if (number > 0) core.setBlock(number, ex, ey);
     }
-    this.insertAction(ev, ex, ey, callback);
-}
+  }
+  this.insertAction(ev, ex, ey, callback);
+};
 
 events.prototype._sys_custom = function (data, callback) {
-    core.insertAction(["请使用\r[yellow]core.registerSystemEvent('custom', func)\r来处理自己添加的系统触发器!"],
-        data.x, data.y, callback);
-}
+  core.insertAction(
+    [
+      "请使用\r[yellow]core.registerSystemEvent('custom', func)\r来处理自己添加的系统触发器!",
+    ],
+    data.x,
+    data.y,
+    callback
+  );
+};
 
 // ------ 自定义事件的处理 ------ //
 
@@ -933,2856 +1044,3552 @@ events.prototype._sys_custom = function (data, callback) {
 // type为事件名,func为事件的处理函数,可接受(data, x, y, prefix)参数
 // data为事件内容,x和y为当前点坐标(可为null),prefix为当前点前缀
 events.prototype.registerEvent = function (type, func) {
-    this.actions[type] = func;
-}
+  this.actions[type] = func;
+};
 
 ////// 注销一个自定义事件
 events.prototype.unregisterEvent = function (type) {
-    delete this.actions[type];
-}
+  delete this.actions[type];
+};
 
 ////// 执行一个自定义事件
 events.prototype.doEvent = function (data, x, y, prefix) {
-    var type = data.type;
-    if (this.actions[type]) {
-        try {
-            return core.doFunc(this.actions[type], this, data, x, y, prefix);
-        }
-        catch (e) {
-            console.error(e);
-            console.error("ERROR in actions[" + type + "]");
-        }
+  var type = data.type;
+  if (this.actions[type]) {
+    try {
+      return core.doFunc(this.actions[type], this, data, x, y, prefix);
+    } catch (e) {
+      console.error(e);
+      console.error("ERROR in actions[" + type + "]");
     }
-    if (this["_action_" + type]) return this["_action_" + type](data, x, y, prefix);
-    core.insertAction("未知的自定义事件: " + type + "!");
-    core.doAction();
-}
+  }
+  if (this["_action_" + type])
+    return this["_action_" + type](data, x, y, prefix);
+  core.insertAction("未知的自定义事件: " + type + "!");
+  core.doAction();
+};
 
 events.prototype.setEvents = function (list, x, y, callback) {
-    var data = core.status.event.data || {};
-    if (list) {
-        var l = core.clone(list);
-        if (!(l instanceof Array)) l = [l];
-        l.push({ "type": "_label" });
-        data.list = [{ todo: l, total: core.clone(l), condition: "false" }];
-        // 结束所有正在执行的自动事件
-        if (list.length == 0) {
-            core.status.autoEvents.forEach(function (autoEvent) {
-                core.autoEventExecuting(autoEvent.symbol, false);
-            });
-        }
+  var data = core.status.event.data || {};
+  if (list) {
+    var l = core.clone(list);
+    if (!(l instanceof Array)) l = [l];
+    l.push({ type: "_label" });
+    data.list = [{ todo: l, total: core.clone(l), condition: "false" }];
+    // 结束所有正在执行的自动事件
+    if (list.length == 0) {
+      core.status.autoEvents.forEach(function (autoEvent) {
+        core.autoEventExecuting(autoEvent.symbol, false);
+      });
     }
-    if (x != null) data.x = x;
-    if (y != null) data.y = y;
-    if (callback) data.callback = callback;
-    if (!data.appendingEvents) data.appendingEvents = [];
-    if (!data.locStack) data.locStack = [];
-    core.status.event.id = 'action';
-    core.status.event.data = data;
-}
+  }
+  if (x != null) data.x = x;
+  if (y != null) data.y = y;
+  if (callback) data.callback = callback;
+  if (!data.appendingEvents) data.appendingEvents = [];
+  if (!data.locStack) data.locStack = [];
+  core.status.event.id = "action";
+  core.status.event.data = data;
+};
 
 ////// 开始执行一系列自定义事件 //////
 events.prototype.startEvents = function (list, x, y, callback) {
-    if (!list) return;
-    if (!(list instanceof Array)) {
-        list = [list];
-    }
-    this.setEvents(list, x, y, callback);
-    // 停止勇士
-    core.waitHeroToStop(function () {
-        core.lockControl();
-        core.doAction();
-    });
-}
+  if (!list) return;
+  if (!(list instanceof Array)) {
+    list = [list];
+  }
+  this.setEvents(list, x, y, callback);
+  // 停止勇士
+  core.waitHeroToStop(function () {
+    core.lockControl();
+    core.doAction();
+  });
+};
 
 ////// 执行当前自定义事件列表中的下一个事件 //////
 events.prototype.doAction = function () {
-    // 清空boxAnimate和UI层
-    clearInterval(core.status.event.interval);
-    clearTimeout(core.status.event.interval);
-    clearInterval(core.status.event.animateUI);
-    core.status.event.interval = null;
-    delete core.status.event.aniamteUI;
-    if (core.status.gameOver || core.status.replay.failed) return;
-    // 判定是否执行完毕
-    if (this._doAction_finishEvents()) return;
-    core.clearUI();
-    var floorId = core.status.event.data.floorId || core.status.floorId;
-    // 当前点坐标和前缀
-    var x = core.status.event.data.x, y = core.status.event.data.y;
-    var prefix = [floorId || ":f", x != null ? x : "x", y != null ? y : "y"].join("@");
-    var current = core.status.event.data.list[0];
-    if (this._popEvents(current, prefix)) return;
-    // 当前要执行的事件
-    var data = current.todo.shift();
-    core.status.event.data.current = data;
-    if (typeof data == "string")
-        data = { "type": "text", "text": data };
-    // 该事件块已经被禁用
-    if (data._disabled) return core.doAction();
-    data.floorId = data.floorId || floorId;
-    core.status.event.data.type = data.type;
-    this.doEvent(data, x, y, prefix);
-    return;
-}
+  // 清空boxAnimate和UI层
+  clearInterval(core.status.event.interval);
+  clearTimeout(core.status.event.interval);
+  clearInterval(core.status.event.animateUI);
+  core.status.event.interval = null;
+  delete core.status.event.aniamteUI;
+  if (core.status.gameOver || core.status.replay.failed) return;
+  // 判定是否执行完毕
+  if (this._doAction_finishEvents()) return;
+  core.clearUI();
+  var floorId = core.status.event.data.floorId || core.status.floorId;
+  // 当前点坐标和前缀
+  var x = core.status.event.data.x,
+    y = core.status.event.data.y;
+  var prefix = [floorId || ":f", x != null ? x : "x", y != null ? y : "y"].join(
+    "@"
+  );
+  var current = core.status.event.data.list[0];
+  if (this._popEvents(current, prefix)) return;
+  // 当前要执行的事件
+  var data = current.todo.shift();
+  core.status.event.data.current = data;
+  if (typeof data == "string") data = { type: "text", text: data };
+  // 该事件块已经被禁用
+  if (data._disabled) return core.doAction();
+  data.floorId = data.floorId || floorId;
+  core.status.event.data.type = data.type;
+  this.doEvent(data, x, y, prefix);
+  return;
+};
 
 events.prototype._doAction_finishEvents = function () {
-    if (core.status.event.id != 'action') return true;
-    // 事件处理完毕
-    if (core.status.event.data.list.length == 0) {
-        // 检测并执行延迟自动事件
-        if (core.status.event.data.appendingEvents.length > 0) {
-            this.setEvents(core.status.event.data.appendingEvents.shift());
-            return false;
-        }
-        var callback = core.status.event.data.callback;
-        core.ui.closePanel();
-        if (callback) callback();
-        core.replay();
-        return true;
+  if (core.status.event.id != "action") return true;
+  // 事件处理完毕
+  if (core.status.event.data.list.length == 0) {
+    // 检测并执行延迟自动事件
+    if (core.status.event.data.appendingEvents.length > 0) {
+      this.setEvents(core.status.event.data.appendingEvents.shift());
+      return false;
     }
-    return false;
-}
+    var callback = core.status.event.data.callback;
+    core.ui.closePanel();
+    if (callback) callback();
+    core.replay();
+    return true;
+  }
+  return false;
+};
 
 events.prototype._popEvents = function (current, prefix) {
-    if (current.todo.length == 0) { // current list is empty
-        if (core.calValue(current.condition, prefix)) { // check condition
-            current.todo = core.clone(current.total);
-        }
-        else {
-            core.status.event.data.list.shift(); // remove stack
-        }
-        core.doAction();
-        return true;
+  if (current.todo.length == 0) {
+    // current list is empty
+    if (core.calValue(current.condition, prefix)) {
+      // check condition
+      current.todo = core.clone(current.total);
+    } else {
+      core.status.event.data.list.shift(); // remove stack
     }
-    return false;
-}
+    core.doAction();
+    return true;
+  }
+  return false;
+};
 
 ////// 往当前事件列表之前或之后添加一个或多个事件 //////
 events.prototype.insertAction = function (action, x, y, callback, addToLast) {
-    if (core.hasFlag("__statistics__")) return;
-    if (core.status.gameOver) return;
-    if (!action) return;
-    core.clearRouteFolding();
+  if (core.hasFlag("__statistics__")) return;
+  if (core.status.gameOver) return;
+  if (!action) return;
+  core.clearRouteFolding();
 
-    action = this.precompile(action);
+  action = this.precompile(action);
 
-    if (core.status.event.id != 'action') {
-        this.startEvents(action, x, y, callback);
-    }
-    else {
-        if (addToLast) {
-            var list = core.status.event.data.list[0].todo;
-            var index = 0;
-            for (var index = 0; index < list.length; index++) {
-                if (list[index].type == '_label') {
-                    list.splice(index, 0, action);
-                    break;
-                }
-            }
+  if (core.status.event.id != "action") {
+    this.startEvents(action, x, y, callback);
+  } else {
+    if (addToLast) {
+      var list = core.status.event.data.list[0].todo;
+      var index = 0;
+      for (var index = 0; index < list.length; index++) {
+        if (list[index].type == "_label") {
+          list.splice(index, 0, action);
+          break;
         }
-        else core.unshift(core.status.event.data.list[0].todo, action);
-        this.setEvents(null, x, y, callback);
-    }
-}
+      }
+    } else core.unshift(core.status.event.data.list[0].todo, action);
+    this.setEvents(null, x, y, callback);
+  }
+};
 
 ////// 往当前事件列表之前或之后添加一个公共事件 //////
-events.prototype.insertCommonEvent = function (name, args, x, y, callback, addToLast) {
-    var commonEvent = this.getCommonEvent(name);
-    if (!commonEvent) {
-        if (callback) callback();
-        return;
-    }
+events.prototype.insertCommonEvent = function (
+  name,
+  args,
+  x,
+  y,
+  callback,
+  addToLast
+) {
+  var commonEvent = this.getCommonEvent(name);
+  if (!commonEvent) {
+    if (callback) callback();
+    return;
+  }
 
-    // 设置参数
-    core.setFlag('arg0', name);
-    if (args instanceof Array) {
-        for (var i = 0; i < args.length; ++i) {
-            try {
-                if (args[i] != null)
-                    core.setFlag('arg' + (i + 1), args[i]);
-            } catch (ee) { console.error(ee) }
-        }
+  // 设置参数
+  core.setFlag("arg0", name);
+  if (args instanceof Array) {
+    for (var i = 0; i < args.length; ++i) {
+      try {
+        if (args[i] != null) core.setFlag("arg" + (i + 1), args[i]);
+      } catch (ee) {
+        console.error(ee);
+      }
     }
+  }
 
-    this.insertAction({ "type": "dowhile", "condition": "false", "data": commonEvent }, x, y, callback, addToLast);
-}
+  this.insertAction(
+    { type: "dowhile", condition: "false", data: commonEvent },
+    x,
+    y,
+    callback,
+    addToLast
+  );
+};
 
 ////// 获得一个公共事件 //////
 events.prototype.getCommonEvent = function (name) {
-    if (!name || typeof name !== 'string') return null;
-    return this.commonEvent[name] || null;
-}
+  if (!name || typeof name !== "string") return null;
+  return this.commonEvent[name] || null;
+};
 
 ////// 恢复一个事件 //////
 events.prototype.recoverEvents = function (data) {
-    if (data) {
-        core.ui.closePanel();
-        core.lockControl();
-        core.status.event.id = 'action';
-        core.status.event.data = data;
-        setTimeout(function () {
-            core.doAction();
-        }, 30);
-        return true;
-    }
-    return false;
-}
+  if (data) {
+    core.ui.closePanel();
+    core.lockControl();
+    core.status.event.id = "action";
+    core.status.event.data = data;
+    setTimeout(function () {
+      core.doAction();
+    }, 30);
+    return true;
+  }
+  return false;
+};
 
 ////// 检测自动事件 //////
 events.prototype.checkAutoEvents = function () {
-    // 只有在无操作或事件流中才能执行自动事件!
-    if (!core.isPlaying() || (core.status.lockControl && core.status.event.id != 'action')) return;
-    if (core.hasFlag('__doNotCheckAutoEvents__')) return;
-    var todo = [], delay = [];
-    core.status.autoEvents.forEach(function (autoEvent) {
-        var symbol = autoEvent.symbol, x = autoEvent.x, y = autoEvent.y, floorId = autoEvent.floorId;
-        // 不在当前楼层 or 已经执行过 or 已被分区 or 正在执行中
-        if (autoEvent.currentFloor && floorId != core.status.floorId) return;
-        if (!autoEvent.multiExecute && core.autoEventExecuted(symbol)) return;
-        if ((flags.__removed__ || []).indexOf(floorId) >= 0) return;
-        if (core.autoEventExecuting(symbol)) return;
-        var prefix = floorId + "@" + x + "@" + y;
-        try {
-            if (!core.calValue(autoEvent.condition, prefix)) return;
-        } catch (e) {
-            return;
-        }
-
-        core.autoEventExecuting(symbol, true);
-        core.autoEventExecuted(symbol, true);
-
-        var event;
-        if (x == null && y == null) {
-            event = [
-                // 用do-while(0)包一层防止break影响事件流
-                { "type": "dowhile", "condition": "false", "data": autoEvent.data },
-                {
-                    "type": "function", "function":
-                        "function() { core.autoEventExecuting('" + symbol + "', false); }"
-                }
-            ];
-        } else {
-            event = [
-                {
-                    "type": "function", "function":
-                        "function() { core.pushEventLoc(" + x + ", " + y + ", '" + floorId + "' ); }"
-                },
-                // 用do-while(0)包一层防止break影响事件流
-                { "type": "dowhile", "condition": "false", "data": autoEvent.data },
-                {
-                    "type": "function", "function":
-                        "function() { core.popEventLoc(); core.autoEventExecuting('" + symbol + "', false); }"
-                }
-            ];
-        }
-
-        if (autoEvent.delayExecute)
-            delay.push(event);
-        else
-            core.push(todo, event);
-    });
-
-    if (todo.length == 0 && delay.length == 0) return;
-
-    if (core.status.event.id == 'action' || todo.length > 0) {
-        core.insertAction(todo);
-        core.push(core.status.event.data.appendingEvents, delay);
-    } else {
-        core.insertAction(delay[0]);
-        if (delay.length > 0) {
-            core.insertAction(delay.slice(1));
-        }
+  // 只有在无操作或事件流中才能执行自动事件!
+  if (
+    !core.isPlaying() ||
+    (core.status.lockControl && core.status.event.id != "action")
+  )
+    return;
+  if (core.hasFlag("__doNotCheckAutoEvents__")) return;
+  var todo = [],
+    delay = [];
+  core.status.autoEvents.forEach(function (autoEvent) {
+    var symbol = autoEvent.symbol,
+      x = autoEvent.x,
+      y = autoEvent.y,
+      floorId = autoEvent.floorId;
+    // 不在当前楼层 or 已经执行过 or 已被分区 or 正在执行中
+    if (autoEvent.currentFloor && floorId != core.status.floorId) return;
+    if (!autoEvent.multiExecute && core.autoEventExecuted(symbol)) return;
+    if ((flags.__removed__ || []).indexOf(floorId) >= 0) return;
+    if (core.autoEventExecuting(symbol)) return;
+    var prefix = floorId + "@" + x + "@" + y;
+    try {
+      if (!core.calValue(autoEvent.condition, prefix)) return;
+    } catch (e) {
+      return;
     }
 
-}
+    core.autoEventExecuting(symbol, true);
+    core.autoEventExecuted(symbol, true);
+
+    var event;
+    if (x == null && y == null) {
+      event = [
+        // 用do-while(0)包一层防止break影响事件流
+        { type: "dowhile", condition: "false", data: autoEvent.data },
+        {
+          type: "function",
+          function:
+            "function() { core.autoEventExecuting('" + symbol + "', false); }",
+        },
+      ];
+    } else {
+      event = [
+        {
+          type: "function",
+          function:
+            "function() { core.pushEventLoc(" +
+            x +
+            ", " +
+            y +
+            ", '" +
+            floorId +
+            "' ); }",
+        },
+        // 用do-while(0)包一层防止break影响事件流
+        { type: "dowhile", condition: "false", data: autoEvent.data },
+        {
+          type: "function",
+          function:
+            "function() { core.popEventLoc(); core.autoEventExecuting('" +
+            symbol +
+            "', false); }",
+        },
+      ];
+    }
+
+    if (autoEvent.delayExecute) delay.push(event);
+    else core.push(todo, event);
+  });
+
+  if (todo.length == 0 && delay.length == 0) return;
+
+  if (core.status.event.id == "action" || todo.length > 0) {
+    core.insertAction(todo);
+    core.push(core.status.event.data.appendingEvents, delay);
+  } else {
+    core.insertAction(delay[0]);
+    if (delay.length > 0) {
+      core.insertAction(delay.slice(1));
+    }
+  }
+};
 
 events.prototype.autoEventExecuting = function (symbol, value) {
-    var aei = core.getFlag('__aei__', []);
-    if (value == null) return aei.indexOf(symbol) >= 0;
-    else {
-        aei = aei.filter(function (one) { return one != symbol; });
-        if (value) aei.push(symbol);
-        core.setFlag('__aei__', aei);
-    }
-}
+  var aei = core.getFlag("__aei__", []);
+  if (value == null) return aei.indexOf(symbol) >= 0;
+  else {
+    aei = aei.filter(function (one) {
+      return one != symbol;
+    });
+    if (value) aei.push(symbol);
+    core.setFlag("__aei__", aei);
+  }
+};
 
 events.prototype.autoEventExecuted = function (symbol, value) {
-    var aed = core.getFlag('__aed__', []);
-    if (value == null) return aed.indexOf(symbol) >= 0;
-    else {
-        aed = aed.filter(function (one) { return one != symbol; });
-        if (value) aed.push(symbol);
-        core.setFlag('__aed__', aed);
-    }
-}
+  var aed = core.getFlag("__aed__", []);
+  if (value == null) return aed.indexOf(symbol) >= 0;
+  else {
+    aed = aed.filter(function (one) {
+      return one != symbol;
+    });
+    if (value) aed.push(symbol);
+    core.setFlag("__aed__", aed);
+  }
+};
 
 events.prototype.pushEventLoc = function (x, y, floorId) {
-    if (core.status.event.id != 'action') return;
-    core.status.event.data.locStack.push({
-        x: core.status.event.data.x,
-        y: core.status.event.data.y,
-        floorId: core.status.event.data.floorId
-    });
-    core.status.event.data.x = x;
-    core.status.event.data.y = y;
-    core.status.event.data.floorId = floorId;
-}
+  if (core.status.event.id != "action") return;
+  core.status.event.data.locStack.push({
+    x: core.status.event.data.x,
+    y: core.status.event.data.y,
+    floorId: core.status.event.data.floorId,
+  });
+  core.status.event.data.x = x;
+  core.status.event.data.y = y;
+  core.status.event.data.floorId = floorId;
+};
 
 events.prototype.popEventLoc = function () {
-    if (core.status.event.id != 'action') return;
-    var loc = core.status.event.data.locStack.shift();
-    if (loc) {
-        core.status.event.data.x = loc.x;
-        core.status.event.data.y = loc.y;
-        core.status.event.data.floorId = loc.floorId;
-    }
-}
+  if (core.status.event.id != "action") return;
+  var loc = core.status.event.data.locStack.shift();
+  if (loc) {
+    core.status.event.data.x = loc.x;
+    core.status.event.data.y = loc.y;
+    core.status.event.data.floorId = loc.floorId;
+  }
+};
 
 events.prototype.precompile = function (data) {
-    var array = this.__precompile_getArray();
-    if (typeof data == 'string') {
-        return this.__precompile_text(data);
-    }
-    if (data instanceof Array) {
-        for (var i = 0; i < data.length; ++i) {
-            data[i] = this.precompile(data[i]);
-        }
-        return data;
-    }
-    if (data && data.type) {
-        if (this["_precompile_" + data.type]) {
-            data = this["_precompile_" + data.type](data);
-        }
-        if (array.texts.indexOf(data.type) >= 0) {
-            data.text = this.__precompile_text(data.text);
-        }
-        if (array.locs.indexOf(data.type) >= 0) {
-            data.loc = this.__precompile_array(data.loc);
-        }
-        if (array.values.indexOf(data.type) >= 0) {
-            data.value = core.replaceValue(data.value);
-        }
-        if (array.uievents.indexOf(data.type) >= 0) {
-            data.x = core.replaceValue(data.x);
-            data.y = core.replaceValue(data.y);
-            data.width = core.replaceValue(data.width);
-            data.height = core.replaceValue(data.height);
-        }
-        if (data.type in array.others) {
-            array.others[data.type].forEach(function (field) {
-                data[field] = core.replaceValue(data[field]);
-            })
-        }
+  var array = this.__precompile_getArray();
+  if (typeof data == "string") {
+    return this.__precompile_text(data);
+  }
+  if (data instanceof Array) {
+    for (var i = 0; i < data.length; ++i) {
+      data[i] = this.precompile(data[i]);
     }
     return data;
-}
+  }
+  if (data && data.type) {
+    if (this["_precompile_" + data.type]) {
+      data = this["_precompile_" + data.type](data);
+    }
+    if (array.texts.indexOf(data.type) >= 0) {
+      data.text = this.__precompile_text(data.text);
+    }
+    if (array.locs.indexOf(data.type) >= 0) {
+      data.loc = this.__precompile_array(data.loc);
+    }
+    if (array.values.indexOf(data.type) >= 0) {
+      data.value = core.replaceValue(data.value);
+    }
+    if (array.uievents.indexOf(data.type) >= 0) {
+      data.x = core.replaceValue(data.x);
+      data.y = core.replaceValue(data.y);
+      data.width = core.replaceValue(data.width);
+      data.height = core.replaceValue(data.height);
+    }
+    if (data.type in array.others) {
+      array.others[data.type].forEach(function (field) {
+        data[field] = core.replaceValue(data[field]);
+      });
+    }
+  }
+  return data;
+};
 
 events.prototype.__precompile_getArray = function () {
-    var texts = [
-        "text", "autoText", "scrollText", "tip", "textImage", "input", "input2",
-        "choices", "confirm", "fillText", "fillBoldText", "drawTextContent"
-    ];
-    var locs = [
-        "show", "hide", "setBlock", "setBlockOpacity", "showFloorImg", "hideFloorImg", "showBgFgMap",
-        "hideBgFgMap", "setBgFgBlock", "animate", "setViewport", "move", "jumoHero",
-        "changeFloor", "changePos", "showTextImage", "showGif", "openDoor",
-        "closeDoor", "battle", "trigger", "insert", "setEnemyOnPoint", "resetEnemyOnPoint"
-    ];
-    var values = [
-        "setValue", "setEnemy", "setEnemyOnPoint", "setEquip", "setFloor", "setGlobalValue",
-    ];
-    var uievents = [
-        "clearMap", "fillText", "fillBoldText", "fillRect", "strokeRect", "fillEllipse", "strokeEllipse",
-        "fillArc", "strokeArc", "drawIcon", "drawSelector", "drawBackground",
-    ];
-    var others = {
-        "fillEllipse": ["a", "b", "angle"],
-        "strokeEllipse": ["a", "b", "angle"],
-        "fillRect": ["radius", "angle"],
-        "strokeRect": ["radius", "angle"],
-        "fillArc": ["r", "start", "end"],
-        "strokeArc": ["r", "start", "end"],
-        "drawLine": ["x1", "y1", "x2", "y2"],
-        "drawArrow": ["x1", "y1", "x2", "y2"],
-        "drawImage": ["x", "y", "w", "h", "x1", "y1", "w1", "h1", "angle"],
-        "drawTextContent": ["left", "top"],
-    };
-    return {
-        texts: texts,
-        locs: locs,
-        values: values,
-        uievents: uievents,
-        others: others
-    };
-}
+  var texts = [
+    "text",
+    "autoText",
+    "scrollText",
+    "tip",
+    "textImage",
+    "input",
+    "input2",
+    "choices",
+    "confirm",
+    "fillText",
+    "fillBoldText",
+    "drawTextContent",
+  ];
+  var locs = [
+    "show",
+    "hide",
+    "setBlock",
+    "setBlockOpacity",
+    "showFloorImg",
+    "hideFloorImg",
+    "showBgFgMap",
+    "hideBgFgMap",
+    "setBgFgBlock",
+    "animate",
+    "setViewport",
+    "move",
+    "jumoHero",
+    "changeFloor",
+    "changePos",
+    "showTextImage",
+    "showGif",
+    "openDoor",
+    "closeDoor",
+    "battle",
+    "trigger",
+    "insert",
+    "setEnemyOnPoint",
+    "resetEnemyOnPoint",
+  ];
+  var values = [
+    "setValue",
+    "setEnemy",
+    "setEnemyOnPoint",
+    "setEquip",
+    "setFloor",
+    "setGlobalValue",
+  ];
+  var uievents = [
+    "clearMap",
+    "fillText",
+    "fillBoldText",
+    "fillRect",
+    "strokeRect",
+    "fillEllipse",
+    "strokeEllipse",
+    "fillArc",
+    "strokeArc",
+    "drawIcon",
+    "drawSelector",
+    "drawBackground",
+  ];
+  var others = {
+    fillEllipse: ["a", "b", "angle"],
+    strokeEllipse: ["a", "b", "angle"],
+    fillRect: ["radius", "angle"],
+    strokeRect: ["radius", "angle"],
+    fillArc: ["r", "start", "end"],
+    strokeArc: ["r", "start", "end"],
+    drawLine: ["x1", "y1", "x2", "y2"],
+    drawArrow: ["x1", "y1", "x2", "y2"],
+    drawImage: ["x", "y", "w", "h", "x1", "y1", "w1", "h1", "angle"],
+    drawTextContent: ["left", "top"],
+  };
+  return {
+    texts: texts,
+    locs: locs,
+    values: values,
+    uievents: uievents,
+    others: others,
+  };
+};
 
 events.prototype.__precompile_text = function (text) {
-    if (typeof text != 'string') return text;
-    return text.replace(/\${(.*?)}/g, function (word, value) {
-        return "${" + core.replaceValue(value) + "}";
-    });
-}
+  if (typeof text != "string") return text;
+  return text.replace(/\${(.*?)}/g, function (word, value) {
+    return "${" + core.replaceValue(value) + "}";
+  });
+};
 
 events.prototype.__precompile_array = function (value) {
-    if (typeof value == 'string') {
-        value = core.replaceValue(value);
-        return value;
-    }
-    if (value instanceof Array) {
-        for (var i = 0; i < value.length; ++i) {
-            value[i] = this.__precompile_array(value[i]);
-        }
-    }
+  if (typeof value == "string") {
+    value = core.replaceValue(value);
     return value;
-}
+  }
+  if (value instanceof Array) {
+    for (var i = 0; i < value.length; ++i) {
+      value[i] = this.__precompile_array(value[i]);
+    }
+  }
+  return value;
+};
 
 // ------ 样板提供的的自定义事件 ------ //
 
 events.prototype.__action_checkReplaying = function () {
-    if (core.isReplaying()) {
-        core.doAction();
-        return true;
-    }
-    return false;
-}
+  if (core.isReplaying()) {
+    core.doAction();
+    return true;
+  }
+  return false;
+};
 
 events.prototype.__action_getLoc = function (loc, x, y, prefix) {
-    if (loc) {
-        x = core.calValue(loc[0], prefix);
-        y = core.calValue(loc[1], prefix);
-    }
-    return [x, y];
-}
+  if (loc) {
+    x = core.calValue(loc[0], prefix);
+    y = core.calValue(loc[1], prefix);
+  }
+  return [x, y];
+};
 
 events.prototype.__action_getHeroLoc = function (loc, prefix) {
-    return this.__action_getLoc(loc, core.getHeroLoc('x'), core.getHeroLoc('y'), prefix);
-}
+  return this.__action_getLoc(
+    loc,
+    core.getHeroLoc("x"),
+    core.getHeroLoc("y"),
+    prefix
+  );
+};
 
 events.prototype.__action_getLoc2D = function (loc, x, y, prefix) {
-    if (!(loc && loc[0] instanceof Array))
-        loc = [this.__action_getLoc(loc, x, y, prefix)];
-    return loc;
-}
+  if (!(loc && loc[0] instanceof Array))
+    loc = [this.__action_getLoc(loc, x, y, prefix)];
+  return loc;
+};
 
 events.prototype.__action_doAsyncFunc = function (isAsync, func) {
-    var parameters = Array.prototype.slice.call(arguments, 2);
-    if (isAsync) {
-        func.apply(this, parameters);
-        core.doAction();
-    }
-    else {
-        func.apply(this, parameters.concat(core.doAction));
-    }
-}
+  var parameters = Array.prototype.slice.call(arguments, 2);
+  if (isAsync) {
+    func.apply(this, parameters);
+    core.doAction();
+  } else {
+    func.apply(this, parameters.concat(core.doAction));
+  }
+};
 
 events.prototype._action_text = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    data.text = core.replaceText(data.text, prefix);
-    var ctx = data.code ? ('__text__' + data.code) : null;;
-    data.ctx = ctx;
-    if (core.getContextByName(ctx) && !data.showAll) {
-        core.ui._animateUI('hide', ctx, function () {
-            core.ui.drawTextBox(data.text, data);
-            core.ui._animateUI('show', ctx, function () {
-                if (data.async) core.doAction();
-            });
-        });
-        return;
-    }
-    core.ui.drawTextBox(data.text, data);
-    if (!data.showAll) {
-        core.ui._animateUI('show', ctx, function () {
-            if (data.async) core.doAction();
-        });
-    }
-}
+  if (this.__action_checkReplaying()) return;
+  data.text = core.replaceText(data.text, prefix);
+  var ctx = data.code ? "__text__" + data.code : null;
+  data.ctx = ctx;
+  if (core.getContextByName(ctx) && !data.showAll) {
+    core.ui._animateUI("hide", ctx, function () {
+      core.ui.drawTextBox(data.text, data);
+      core.ui._animateUI("show", ctx, function () {
+        if (data.async) core.doAction();
+      });
+    });
+    return;
+  }
+  core.ui.drawTextBox(data.text, data);
+  if (!data.showAll) {
+    core.ui._animateUI("show", ctx, function () {
+      if (data.async) core.doAction();
+    });
+  }
+};
 
 events.prototype._action_moveTextBox = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    this.__action_doAsyncFunc(data.async, core.moveTextBox,
-        data.code, this.__action_getLoc(data.loc, x, y, prefix), data.relative, data.moveMode, data.time);
-}
+  if (this.__action_checkReplaying()) return;
+  this.__action_doAsyncFunc(
+    data.async,
+    core.moveTextBox,
+    data.code,
+    this.__action_getLoc(data.loc, x, y, prefix),
+    data.relative,
+    data.moveMode,
+    data.time
+  );
+};
 
 events.prototype._action_clearTextBox = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    core.clearTextBox(data.code, core.doAction);
-}
+  if (this.__action_checkReplaying()) return;
+  core.clearTextBox(data.code, core.doAction);
+};
 
 events.prototype._action_autoText = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    data.text = core.replaceText(data.text, prefix);
-    core.ui.drawTextBox(data.text);
-    setTimeout(core.doAction, data.time || 3000);
-}
+  if (this.__action_checkReplaying()) return;
+  data.text = core.replaceText(data.text, prefix);
+  core.ui.drawTextBox(data.text);
+  setTimeout(core.doAction, data.time || 3000);
+};
 
 events.prototype._action_scrollText = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    data.text = core.replaceText(data.text, prefix);
-    this.__action_doAsyncFunc(data.async, core.drawScrollText, data.text, data.time || 5000, data.lineHeight || 1.4);
-}
+  if (this.__action_checkReplaying()) return;
+  data.text = core.replaceText(data.text, prefix);
+  this.__action_doAsyncFunc(
+    data.async,
+    core.drawScrollText,
+    data.text,
+    data.time || 5000,
+    data.lineHeight || 1.4
+  );
+};
 
 events.prototype._action_comment = function (data, x, y, prefix) {
-    core.doAction();
-}
+  core.doAction();
+};
 
 events.prototype._action__label = function (data, x, y, prefix) {
-    core.doAction();
-}
+  core.doAction();
+};
 
 events.prototype._action_setText = function (data, x, y, prefix) {
-    this.setTextAttribute(data);
-    core.doAction();
-}
+  this.setTextAttribute(data);
+  core.doAction();
+};
 
 events.prototype._action_tip = function (data, x, y, prefix) {
-    core.drawTip(core.replaceText(data.text, prefix), data.icon);
-    core.doAction();
-}
+  core.drawTip(core.replaceText(data.text, prefix), data.icon);
+  core.doAction();
+};
 
 events.prototype._action_show = function (data, x, y, prefix) {
-    data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    if (data.time > 0 && data.floorId == core.status.floorId) {
-        this.__action_doAsyncFunc(data.async, core.animateBlock, data.loc, 'show', data.time);
-    }
-    else {
-        data.loc.forEach(function (t) {
-            core.showBlock(t[0], t[1], data.floorId);
-        });
-        core.doAction();
-    }
-}
+  data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  if (data.time > 0 && data.floorId == core.status.floorId) {
+    this.__action_doAsyncFunc(
+      data.async,
+      core.animateBlock,
+      data.loc,
+      "show",
+      data.time
+    );
+  } else {
+    data.loc.forEach(function (t) {
+      core.showBlock(t[0], t[1], data.floorId);
+    });
+    core.doAction();
+  }
+};
 
 events.prototype._action_hide = function (data, x, y, prefix) {
-    data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    if (data.time > 0 && data.floorId == core.status.floorId) {
-        this.__action_doAsyncFunc(data.async, core.animateBlock, data.loc, data.remove ? 'remove' : 'hide', data.time);
-    }
-    else {
-        data.loc.forEach(function (t) {
-            if (data.remove) core.removeBlock(t[0], t[1], data.floorId);
-            else core.hideBlock(t[0], t[1], data.floorId);
-        });
-        core.doAction();
-    }
-}
+  data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  if (data.time > 0 && data.floorId == core.status.floorId) {
+    this.__action_doAsyncFunc(
+      data.async,
+      core.animateBlock,
+      data.loc,
+      data.remove ? "remove" : "hide",
+      data.time
+    );
+  } else {
+    data.loc.forEach(function (t) {
+      if (data.remove) core.removeBlock(t[0], t[1], data.floorId);
+      else core.hideBlock(t[0], t[1], data.floorId);
+    });
+    core.doAction();
+  }
+};
 
 events.prototype._action_setBlock = function (data, x, y, prefix) {
-    data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    data.time = data.time || 0;
-    data.floorId = data.floorId || core.status.floorId;
-    if (data.time > 0 && data.floorId == core.status.floorId) {
-        this.__action_doAsyncFunc(data.async, core.animateSetBlocks, data.number, data.loc, data.floorId, data.time);
-    } else {
-        data.loc.forEach(function (loc) {
-            core.setBlock(data.number, loc[0], loc[1], data.floorId);
-        });
-        core.doAction();
-    }
-}
+  data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  data.time = data.time || 0;
+  data.floorId = data.floorId || core.status.floorId;
+  if (data.time > 0 && data.floorId == core.status.floorId) {
+    this.__action_doAsyncFunc(
+      data.async,
+      core.animateSetBlocks,
+      data.number,
+      data.loc,
+      data.floorId,
+      data.time
+    );
+  } else {
+    data.loc.forEach(function (loc) {
+      core.setBlock(data.number, loc[0], loc[1], data.floorId);
+    });
+    core.doAction();
+  }
+};
 
 events.prototype._action_setBlockOpacity = function (data, x, y, prefix) {
-    data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    if (data.time > 0 && data.floorId == core.status.floorId) {
-        this.__action_doAsyncFunc(data.async, core.animateBlock, data.loc, data.opacity, data.time);
-    }
-    else {
-        data.loc.forEach(function (t) {
-            core.setBlockOpacity(data.opacity, t[0], t[1], data.floorId);
-        });
-        core.doAction();
-    }
-}
+  data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  if (data.time > 0 && data.floorId == core.status.floorId) {
+    this.__action_doAsyncFunc(
+      data.async,
+      core.animateBlock,
+      data.loc,
+      data.opacity,
+      data.time
+    );
+  } else {
+    data.loc.forEach(function (t) {
+      core.setBlockOpacity(data.opacity, t[0], t[1], data.floorId);
+    });
+    core.doAction();
+  }
+};
 
 events.prototype._action_setBlockFilter = function (data, x, y, prefix) {
-    data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    data.loc.forEach(function (t) {
-        core.setBlockFilter(data, t[0], t[1], data.floorId);
-    });
-    core.doAction();
-}
+  data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  data.loc.forEach(function (t) {
+    core.setBlockFilter(data, t[0], t[1], data.floorId);
+  });
+  core.doAction();
+};
 
 events.prototype._action_turnBlock = function (data, x, y, prefix) {
-    data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    data.loc.forEach(function (t) {
-        core.turnBlock(data.direction, t[0], t[1], data.floorId);
-    });
-    core.doAction();
-}
+  data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  data.loc.forEach(function (t) {
+    core.turnBlock(data.direction, t[0], t[1], data.floorId);
+  });
+  core.doAction();
+};
 
 events.prototype._action_showFloorImg = function (data, x, y, prefix) {
-    core.maps.showFloorImage(this.__action_getLoc2D(data.loc, x, y, prefix), data.floorId, core.doAction);
-}
+  core.maps.showFloorImage(
+    this.__action_getLoc2D(data.loc, x, y, prefix),
+    data.floorId,
+    core.doAction
+  );
+};
 
 events.prototype._action_hideFloorImg = function (data, x, y, prefix) {
-    core.maps.hideFloorImage(this.__action_getLoc2D(data.loc, x, y, prefix), data.floorId, core.doAction);
-}
+  core.maps.hideFloorImage(
+    this.__action_getLoc2D(data.loc, x, y, prefix),
+    data.floorId,
+    core.doAction
+  );
+};
 
 events.prototype._action_showBgFgMap = function (data, x, y, prefix) {
-    core.maps.showBgFgMap(data.name, this.__action_getLoc2D(data.loc, x, y, prefix), data.floorId, core.doAction)
-}
+  core.maps.showBgFgMap(
+    data.name,
+    this.__action_getLoc2D(data.loc, x, y, prefix),
+    data.floorId,
+    core.doAction
+  );
+};
 
 events.prototype._action_hideBgFgMap = function (data, x, y, prefix) {
-    core.maps.hideBgFgMap(data.name, this.__action_getLoc2D(data.loc, x, y, prefix), data.floorId, core.doAction);
-}
+  core.maps.hideBgFgMap(
+    data.name,
+    this.__action_getLoc2D(data.loc, x, y, prefix),
+    data.floorId,
+    core.doAction
+  );
+};
 
 events.prototype._action_setBgFgBlock = function (data, x, y, prefix) {
-    data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    data.loc.forEach(function (t) {
-        core.setBgFgBlock(data.name, data.number, t[0], t[1], data.floorId);
-    });
-    core.doAction();
-}
+  data.loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  data.loc.forEach(function (t) {
+    core.setBgFgBlock(data.name, data.number, t[0], t[1], data.floorId);
+  });
+  core.doAction();
+};
 
 events.prototype._action_follow = function (data, x, y, prefix) {
-    this.follow(data.name);
-    core.doAction();
-}
+  this.follow(data.name);
+  core.doAction();
+};
 
 events.prototype._action_unfollow = function (data, x, y, prefix) {
-    this.unfollow(data.name);
-    core.doAction();
-}
+  this.unfollow(data.name);
+  core.doAction();
+};
 
 events.prototype._action_animate = function (data, x, y, prefix) {
-    if (data.loc == 'hero') {
-        this.__action_doAsyncFunc(data.async, core.drawHeroAnimate, data.name);
-    } else {
-        data.loc = this.__action_getLoc(data.loc, x, y, prefix);
-        this.__action_doAsyncFunc(data.async, core.drawAnimate, data.name, data.loc[0], data.loc[1], data.alignWindow);
-    }
-}
+  if (data.loc == "hero") {
+    this.__action_doAsyncFunc(data.async, core.drawHeroAnimate, data.name);
+  } else {
+    data.loc = this.__action_getLoc(data.loc, x, y, prefix);
+    this.__action_doAsyncFunc(
+      data.async,
+      core.drawAnimate,
+      data.name,
+      data.loc[0],
+      data.loc[1],
+      data.alignWindow
+    );
+  }
+};
 
 events.prototype._action_stopAnimate = function (data, x, y, prefix) {
-    core.stopAnimate(null, data.doCallback);
-    core.doAction();
-}
+  core.stopAnimate(data.id, data.doCallback);
+  core.doAction();
+};
 
 events.prototype._action_setViewport = function (data, x, y, prefix) {
-    if (data.dxy != null) {
-        data.loc = [core.bigmap.offsetX / 32 + (core.calValue(data.dxy[0], prefix) || 0), core.bigmap.offsetY / 32 + (core.calValue(data.dxy[1], prefix) || 0)];
-    } else if (data.loc == null) {
-        data.loc = [core.getHeroLoc('x') - core._HALF_WIDTH_, core.getHeroLoc('y') - core._HALF_HEIGHT_];
-    } else {
-        data.loc = this.__action_getLoc(data.loc, x, y, prefix);
-    }
-    this.__action_doAsyncFunc(data.async, core.moveViewport, data.loc[0], data.loc[1], data.moveMode, data.time);
-}
+  if (data.dxy != null) {
+    data.loc = [
+      core.bigmap.offsetX / 32 + (core.calValue(data.dxy[0], prefix) || 0),
+      core.bigmap.offsetY / 32 + (core.calValue(data.dxy[1], prefix) || 0),
+    ];
+  } else if (data.loc == null) {
+    data.loc = [
+      core.getHeroLoc("x") - core._HALF_WIDTH_,
+      core.getHeroLoc("y") - core._HALF_HEIGHT_,
+    ];
+  } else {
+    data.loc = this.__action_getLoc(data.loc, x, y, prefix);
+  }
+  this.__action_doAsyncFunc(
+    data.async,
+    core.moveViewport,
+    data.loc[0],
+    data.loc[1],
+    data.moveMode,
+    data.time
+  );
+};
 
 events.prototype._action_lockViewport = function (data, x, y, prefix) {
-    core.setFlag('__lockViewport__', data.lock || null);
-    core.doAction();
-}
+  core.setFlag("__lockViewport__", data.lock || null);
+  core.doAction();
+};
 
 events.prototype._action_move = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc(data.loc, x, y, prefix);
-    this.__action_doAsyncFunc(data.async, core.moveBlock, loc[0], loc[1], data.steps, data.time, data.keep);
-}
+  var loc = this.__action_getLoc(data.loc, x, y, prefix);
+  this.__action_doAsyncFunc(
+    data.async,
+    core.moveBlock,
+    loc[0],
+    loc[1],
+    data.steps,
+    data.time,
+    data.keep
+  );
+};
 
 events.prototype._action_moveAction = function (data, x, y, prefix) {
-    // 检查下一个点是否可通行
-    if (core.canMoveHero()) {
-        var nx = core.nextX(), ny = core.nextY();
-        // 检查noPass决定是撞击还是移动
-        if (core.noPass(nx, ny)) {
-            core.insertAction([
-                { "type": "trigger", "loc": [nx, ny] }
-            ]);
-        } else {
-            // 先移动一格,然后尝试触发事件
-            core.insertAction([
-                { "type": "moveHero", "steps": ["forward"] },
-                { "type": "function", "function": "function() { core.moveOneStep(core.doAction); }", "async": true },
-                { "type": "_label" },
-            ]);
-        }
+  // 检查下一个点是否可通行
+  if (core.canMoveHero()) {
+    var nx = core.nextX(),
+      ny = core.nextY();
+    // 检查noPass决定是撞击还是移动
+    if (core.noPass(nx, ny)) {
+      core.insertAction([{ type: "trigger", loc: [nx, ny] }]);
+    } else {
+      // 先移动一格,然后尝试触发事件
+      core.insertAction([
+        { type: "moveHero", steps: ["forward"] },
+        {
+          type: "function",
+          function: "function() { core.moveOneStep(core.doAction); }",
+          async: true,
+        },
+        { type: "_label" },
+      ]);
     }
-    core.doAction();
-}
+  }
+  core.doAction();
+};
 
 events.prototype._action_moveHero = function (data, x, y, prefix) {
-    this.__action_doAsyncFunc(data.async, core.eventMoveHero, data.steps, data.time);
-}
+  this.__action_doAsyncFunc(
+    data.async,
+    core.eventMoveHero,
+    data.steps,
+    data.time
+  );
+};
 
 events.prototype._action_jump = function (data, x, y, prefix) {
-    var from = this.__action_getLoc(data.from, x, y, prefix), to;
-    if (data.dxy) {
-        to = [from[0] + (core.calValue(data.dxy[0], prefix) || 0), from[1] + (core.calValue(data.dxy[1], prefix) || 0)];
-    } else {
-        to = this.__action_getLoc(data.to, x, y, prefix);
-    }
-    this.__action_doAsyncFunc(data.async, core.jumpBlock, from[0], from[1], to[0], to[1], data.time, data.keep);
-}
+  var from = this.__action_getLoc(data.from, x, y, prefix),
+    to;
+  if (data.dxy) {
+    to = [
+      from[0] + (core.calValue(data.dxy[0], prefix) || 0),
+      from[1] + (core.calValue(data.dxy[1], prefix) || 0),
+    ];
+  } else {
+    to = this.__action_getLoc(data.to, x, y, prefix);
+  }
+  this.__action_doAsyncFunc(
+    data.async,
+    core.jumpBlock,
+    from[0],
+    from[1],
+    to[0],
+    to[1],
+    data.time,
+    data.keep
+  );
+};
 
 events.prototype._precompile_jump = function (data) {
-    data.from = this.__precompile_array(data.from);
-    data.to = this.__precompile_array(data.to);
-    data.dxy = this.__precompile_array(data.dxy);
-    return data;
-}
+  data.from = this.__precompile_array(data.from);
+  data.to = this.__precompile_array(data.to);
+  data.dxy = this.__precompile_array(data.dxy);
+  return data;
+};
 
 events.prototype._action_jumpHero = function (data, x, y, prefix) {
-    var loc;
-    if (data.dxy) {
-        loc = [core.getHeroLoc('x') + (core.calValue(data.dxy[0], prefix) || 0), core.getHeroLoc('y') + (core.calValue(data.dxy[1], prefix) || 0)];
-    } else {
-        loc = this.__action_getHeroLoc(data.loc, prefix);
-    }
-    this.__action_doAsyncFunc(data.async, core.jumpHero, loc[0], loc[1], data.time);
-}
+  var loc;
+  if (data.dxy) {
+    loc = [
+      core.getHeroLoc("x") + (core.calValue(data.dxy[0], prefix) || 0),
+      core.getHeroLoc("y") + (core.calValue(data.dxy[1], prefix) || 0),
+    ];
+  } else {
+    loc = this.__action_getHeroLoc(data.loc, prefix);
+  }
+  this.__action_doAsyncFunc(
+    data.async,
+    core.jumpHero,
+    loc[0],
+    loc[1],
+    data.time
+  );
+};
 
 events.prototype._action_changeFloor = function (data, x, y, prefix) {
-    var loc = this.__action_getHeroLoc(data.loc, prefix);
-    var heroLoc = { x: loc[0], y: loc[1], direction: data.direction };
-    core.changeFloor(data.floorId, data.stair, heroLoc, data.time, core.doAction);
-}
+  var loc = this.__action_getHeroLoc(data.loc, prefix);
+  var heroLoc = { x: loc[0], y: loc[1], direction: data.direction };
+  core.changeFloor(data.floorId, data.stair, heroLoc, data.time, core.doAction);
+};
 
 events.prototype._action_changePos = function (data, x, y, prefix) {
-    core.clearMap('hero');
-    if (!data.loc && data.direction) {
-        core.setHeroLoc('direction', core.turnDirection(data.direction), true);
-        core.drawHero();
-        return core.doAction();
-    }
-
-    var loc = this.__action_getHeroLoc(data.loc, prefix);
-    core.setHeroLoc('x', loc[0]);
-    core.setHeroLoc('y', loc[1]);
-    if (data.direction) core.setHeroLoc('direction', core.turnDirection(data.direction));
+  core.clearMap("hero");
+  if (!data.loc && data.direction) {
+    core.setHeroLoc("direction", core.turnDirection(data.direction), true);
     core.drawHero();
-    core.doAction();
-}
+    return core.doAction();
+  }
+
+  var loc = this.__action_getHeroLoc(data.loc, prefix);
+  core.setHeroLoc("x", loc[0]);
+  core.setHeroLoc("y", loc[1]);
+  if (data.direction)
+    core.setHeroLoc("direction", core.turnDirection(data.direction));
+  core.drawHero();
+  core.doAction();
+};
 
 events.prototype._action_showImage = function (data, x, y, prefix) {
-    if (core.isReplaying()) data.time = 0;
-    this.__action_doAsyncFunc(data.async || data.time == 0, core.showImage,
-        data.code, data.image + (data.reverse || ''), data.sloc, data.loc, data.opacity, data.time);
-}
+  if (core.isReplaying()) data.time = 0;
+  this.__action_doAsyncFunc(
+    data.async || data.time == 0,
+    core.showImage,
+    data.code,
+    data.image + (data.reverse || ""),
+    data.sloc,
+    data.loc,
+    data.opacity,
+    data.time
+  );
+};
 
 events.prototype._precompile_showImage = function (data) {
-    data.sloc = this.__precompile_array(data.sloc);
-    data.loc = this.__precompile_array(data.loc);
-    return data;
-}
+  data.sloc = this.__precompile_array(data.sloc);
+  data.loc = this.__precompile_array(data.loc);
+  return data;
+};
 
 events.prototype._action_showTextImage = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc(data.loc, 0, 0, prefix);
-    if (core.isReplaying()) data.time = 0;
-    data.text = core.replaceText(data.text, prefix);
-    var __tmpName = (Math.random() + "_" + Math.random()).replace(/\./g, "") + ".png";
-    core.material.images.images[__tmpName] = core.ui.textImage(data.text);
-    this.__action_doAsyncFunc(data.async || data.time == 0, core.showImage,
-        data.code, __tmpName + (data.reverse || ""), null, loc, data.opacity, data.time);
-    delete core.material.images.images[__tmpName];
-}
+  var loc = this.__action_getLoc(data.loc, 0, 0, prefix);
+  if (core.isReplaying()) data.time = 0;
+  data.text = core.replaceText(data.text, prefix);
+  var __tmpName =
+    (Math.random() + "_" + Math.random()).replace(/\./g, "") + ".png";
+  core.material.images.images[__tmpName] = core.ui.textImage(data.text);
+  this.__action_doAsyncFunc(
+    data.async || data.time == 0,
+    core.showImage,
+    data.code,
+    __tmpName + (data.reverse || ""),
+    null,
+    loc,
+    data.opacity,
+    data.time
+  );
+  delete core.material.images.images[__tmpName];
+};
 
 events.prototype._action_hideImage = function (data, x, y, prefix) {
-    if (core.isReplaying()) data.time = 0;
-    this.__action_doAsyncFunc(data.async || data.time == 0, core.hideImage, data.code, data.time);
-}
+  if (core.isReplaying()) data.time = 0;
+  this.__action_doAsyncFunc(
+    data.async || data.time == 0,
+    core.hideImage,
+    data.code,
+    data.time
+  );
+};
 
 events.prototype._action_showGif = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc(data.loc, 0, 0, prefix);
-    this.showGif(data.name, loc[0], loc[1]);
-    core.doAction();
-}
+  var loc = this.__action_getLoc(data.loc, 0, 0, prefix);
+  this.showGif(data.name, loc[0], loc[1]);
+  core.doAction();
+};
 
 events.prototype._action_moveImage = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    this.__action_doAsyncFunc(data.async, core.moveImage, data.code, data.to, data.opacity, data.moveMode, data.time);
-}
+  if (this.__action_checkReplaying()) return;
+  this.__action_doAsyncFunc(
+    data.async,
+    core.moveImage,
+    data.code,
+    data.to,
+    data.opacity,
+    data.moveMode,
+    data.time
+  );
+};
 
 events.prototype._precompile_moveImage = function (data) {
-    data.to = this.__precompile_array(data.to);
-    return data;
-}
+  data.to = this.__precompile_array(data.to);
+  return data;
+};
 
 events.prototype._action_rotateImage = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    this.__action_doAsyncFunc(data.async, core.rotateImage, data.code, data.center, data.angle, data.moveMode, data.time);
-}
+  if (this.__action_checkReplaying()) return;
+  this.__action_doAsyncFunc(
+    data.async,
+    core.rotateImage,
+    data.code,
+    data.center,
+    data.angle,
+    data.moveMode,
+    data.time
+  );
+};
 
 events.prototype._precompile_rotateImage = function (data) {
-    data.center = this.__precompile_array(data.center);
-    return data;
-}
+  data.center = this.__precompile_array(data.center);
+  return data;
+};
 
 events.prototype._action_scaleImage = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
-    this.__action_doAsyncFunc(data.async, core.scaleImage, data.code, data.center, data.scale, data.moveMode, data.time);
-}
+  if (this.__action_checkReplaying()) return;
+  this.__action_doAsyncFunc(
+    data.async,
+    core.scaleImage,
+    data.code,
+    data.center,
+    data.scale,
+    data.moveMode,
+    data.time
+  );
+};
 
 events.prototype._precompile_scaleImage = function (data) {
-    data.center = this.__precompile_array(data.center);
-    return data;
-}
+  data.center = this.__precompile_array(data.center);
+  return data;
+};
 
 events.prototype._action_setCurtain = function (data, x, y, prefix) {
-    if (data.async) {
-        core.setCurtain(data.color || core.status.thisMap.color, data.time, data.moveMode);
-        if (data.color == null || data.keep) core.setFlag('__color__', data.color || null);
+  if (data.async) {
+    core.setCurtain(
+      data.color || core.status.thisMap.color,
+      data.time,
+      data.moveMode
+    );
+    if (data.color == null || data.keep)
+      core.setFlag("__color__", data.color || null);
+    core.doAction();
+  } else {
+    core.setCurtain(
+      data.color || core.status.thisMap.color,
+      data.time,
+      data.moveMode,
+      function () {
+        if (data.color == null || data.keep)
+          core.setFlag("__color__", data.color || null);
         core.doAction();
-    }
-    else {
-        core.setCurtain(data.color || core.status.thisMap.color, data.time, data.moveMode, function () {
-            if (data.color == null || data.keep) core.setFlag('__color__', data.color || null);
-            core.doAction();
-        });
-    }
-}
+      }
+    );
+  }
+};
 
 events.prototype._action_screenFlash = function (data, x, y, prefix) {
-    this.__action_doAsyncFunc(data.async, core.screenFlash, data.color, data.time, data.times, data.moveMode);
-}
+  this.__action_doAsyncFunc(
+    data.async,
+    core.screenFlash,
+    data.color,
+    data.time,
+    data.times,
+    data.moveMode
+  );
+};
 
 events.prototype._action_setWeather = function (data, x, y, prefix) {
-    core.setWeather(data.name, data.level);
-    if (data.keep && ['rain', 'snow', 'sun', 'fog', 'cloud'].indexOf(data.name) >= 0)
-        core.setFlag('__weather__', [data.name, data.level]);
-    else core.removeFlag('__weather__');
-    core.doAction();
-}
+  core.setWeather(data.name, data.level);
+  if (
+    data.keep &&
+    ["rain", "snow", "sun", "fog", "cloud"].indexOf(data.name) >= 0
+  )
+    core.setFlag("__weather__", [data.name, data.level]);
+  else core.removeFlag("__weather__");
+  core.doAction();
+};
 
 events.prototype._action_openDoor = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc(data.loc, x, y, prefix);
-    var floorId = data.floorId;
-    if (floorId == core.status.floorId) {
-        this.__action_doAsyncFunc(data.async, core.openDoor, loc[0], loc[1], data.needKey);
-    }
-    else {
-        core.removeBlock(loc[0], loc[1], floorId);
-        core.doAction();
-    }
-}
+  var loc = this.__action_getLoc(data.loc, x, y, prefix);
+  var floorId = data.floorId;
+  if (floorId == core.status.floorId) {
+    this.__action_doAsyncFunc(
+      data.async,
+      core.openDoor,
+      loc[0],
+      loc[1],
+      data.needKey
+    );
+  } else {
+    core.removeBlock(loc[0], loc[1], floorId);
+    core.doAction();
+  }
+};
 
 events.prototype._action_closeDoor = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc(data.loc, x, y, prefix);
-    this.__action_doAsyncFunc(data.async, core.closeDoor, loc[0], loc[1], data.id);
-}
+  var loc = this.__action_getLoc(data.loc, x, y, prefix);
+  this.__action_doAsyncFunc(
+    data.async,
+    core.closeDoor,
+    loc[0],
+    loc[1],
+    data.id
+  );
+};
 
 events.prototype._action_useItem = function (data, x, y, prefix) {
-    // 考虑到可能覆盖楼传事件的问题,这里不对fly进行检查。
-    if (data.id != 'book' && core.canUseItem(data.id)) {
-        core.useItem(data.id, true, core.doAction);
-    }
-    else {
-        core.playSound('操作失败');
-        core.drawTip("当前无法使用" + ((core.material.items[data.id] || {}).name || "未知道具"));
-        core.doAction();
-    }
-}
-
-events.prototype._action_loadEquip = function (data, x, y, prefix) {
-    core.loadEquip(data.id);
-    core.doAction();
-}
-
-events.prototype._action_unloadEquip = function (data, x, y, prefix) {
-    core.unloadEquip(data.pos);
-    core.doAction();
-}
-
-events.prototype._action_openShop = function (data, x, y, prefix) {
-    core.setShopVisited(data.id, true);
-    if (data.open) core.openShop(data.id, true);
-    core.doAction();
-}
-
-events.prototype._action_disableShop = function (data, x, y, prefix) {
-    core.setShopVisited(data.id, false);
-    core.doAction();
-}
-
-events.prototype._action_battle = function (data, x, y, prefix) {
-    if (data.id) {
-        this.battle(data.id, null, null, true, core.doAction);
-    }
-    else {
-        if (data.floorId != core.status.floorId) {
-            core.doAction();
-            return;
-        }
-        var loc = this.__action_getLoc(data.loc, x, y, prefix);
-        this.battle(null, loc[0], loc[1], true, core.doAction);
-    }
-}
-
-events.prototype._action_trigger = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc(data.loc, x, y, prefix);
-    this._trigger_inAction(loc[0], loc[1]);
-}
-
-events.prototype._action_insert = function (data, x, y, prefix) {
-    if (data.name) { // 公共事件
-        core.insertCommonEvent(data.name, data.args);
-    }
-    else {
-        // 设置参数
-        if (data.args instanceof Array) {
-            for (var i = 0; i < data.args.length; ++i) {
-                try {
-                    if (data.args[i] != null)
-                        core.setFlag('arg' + (i + 1), data.args[i]);
-                } catch (ee) { console.error(ee) }
-            }
-        }
-        var loc = this.__action_getLoc(data.loc, x, y, prefix);
-        core.setFlag('arg0', loc);
-        var floorId = data.floorId;
-        var which = data.which || "events";
-        var event = (core.floors[floorId][which] || [])[loc[0] + "," + loc[1]];
-        if (event) this.insertAction(event.data || event);
-    }
-    core.doAction();
-}
-
-events.prototype._action_playBgm = function (data, x, y, prefix) {
-    core.playBgm(data.name, data.startTime || 0);
-    core.setFlag("__bgm__", data.keep ? data.name : null);
-    core.doAction();
-}
-
-events.prototype._action_pauseBgm = function (data, x, y, prefix) {
-    core.pauseBgm();
-    core.doAction();
-}
-
-events.prototype._action_resumeBgm = function (data, x, y, prefix) {
-    core.resumeBgm(data.resume);
-    core.doAction();
-}
-
-events.prototype._action_loadBgm = function (data, x, y, prefix) {
-    core.loadBgm(data.name);
-    core.doAction();
-}
-
-events.prototype._action_freeBgm = function (data, x, y, prefix) {
-    core.freeBgm(data.name);
-    core.doAction();
-}
-
-events.prototype._action_playSound = function (data, x, y, prefix) {
-    if (data.stop) core.stopSound();
-    if (data.sync) {
-        core.playSound(data.name, data.pitch, core.doAction);
-    } else {
-        core.playSound(data.name, data.pitch);
-        core.doAction();
-    }
-}
-
-events.prototype._action_stopSound = function (data, x, y, prefix) {
-    core.stopSound();
-    core.doAction();
-}
-
-events.prototype._action_setVolume = function (data, x, y, prefix) {
-    data.value = core.clamp(parseInt(data.value) / 100, 0, 1);
-    core.setFlag("__volume__", data.value);
-    this.__action_doAsyncFunc(data.async, core.setVolume, data.value, data.time || 0);
-}
-
-events.prototype._action_setBgmSpeed = function (data, x, y, prefix) {
-    core.setBgmSpeed(data.value, data.pitch || false);
-    core.doAction();
-}
-
-events.prototype._action_setValue = function (data, x, y, prefix) {
-    this.setValue(data.name, data.operator, data.value, prefix);
-    if (!data.norefresh) {
-        if (core.status.hero.hp <= 0) {
-            core.status.hero.hp = 0;
-            core.updateStatusBar();
-            core.events.lose();
-        } else {
-            core.updateStatusBar();
-        }
-    }
-    core.doAction();
-}
-
-events.prototype._action_addValue = function (data, x, y, prefix) {
-    data.operator = '+=';
-    this._action_setValue(data, x, y, prefix);
-}
-
-events.prototype._action_setEnemy = function (data, x, y, prefix) {
-    this.setEnemy(data.id, data.name, data.value, data.operator, prefix, data.norefresh);
-    core.doAction();
-}
-
-events.prototype._action_setEnemyOnPoint = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    loc.forEach(function (one) {
-        core.setEnemyOnPoint(one[0], one[1], data.floorId, data.name, data.value, data.operator, prefix, data.norefresh);
-    });
-    core.doAction();
-}
-
-events.prototype._action_resetEnemyOnPoint = function (data, x, y, prefix) {
-    var loc = this.__action_getLoc2D(data.loc, x, y, prefix);
-    loc.forEach(function (one) {
-        core.resetEnemyOnPoint(one[0], one[1], data.floorId, data.norefresh);
-    });
-    core.doAction();
-}
-
-events.prototype._precompile_moveEnemyOnPoint = function (data) {
-    data.from = this.__precompile_array(data.from);
-    data.to = this.__precompile_array(data.to);
-    data.dxy = this.__precompile_array(data.dxy);
-    return data;
-}
-
-events.prototype._action_moveEnemyOnPoint = function (data, x, y, prefix) {
-    var from = this.__action_getLoc(data.from, x, y, prefix), to;
-    if (data.dxy) {
-        to = [from[0] + (core.calValue(data.dxy[0], prefix) || 0), from[1] + (core.calValue(data.dxy[1], prefix) || 0)];
-    } else {
-        to = this.__action_getLoc(data.to, x, y, prefix);
-    }
-    this.moveEnemyOnPoint(from[0], from[1], to[0], to[1], data.floorId, data.norefresh);
-    core.doAction();
-}
-
-events.prototype._action_setEquip = function (data, x, y, prefix) {
-    core.setEquip(data.id, data.valueType, data.name, data.value, data.operator, prefix);
-    core.doAction();
-}
-
-events.prototype._action_setFloor = function (data, x, y, prefix) {
-    this.setFloorInfo(data.name, data.value, data.floorId, prefix);
-    core.doAction();
-}
-
-events.prototype._action_setGlobalAttribute = function (data, x, y, prefix) {
-    this.setGlobalAttribute(data.name, data.value);
-    core.doAction();
-}
-
-events.prototype._action_setGlobalValue = function (data, x, y, prefix) {
-    core.values[data.name] = data.value;
-    core.doAction();
-}
-
-events.prototype._action_setGlobalFlag = function (data, x, y, prefix) {
-    this.setGlobalFlag(data.name, data.value);
-    core.doAction();
-}
-
-events.prototype._action_setNameMap = function (data, x, y, floorId) {
-    this.setNameMap(data.name, data.value);
-    core.doAction();
-}
-
-events.prototype._action_setHeroIcon = function (data, x, y, prefix) {
-    this.setHeroIcon(data.name, data.noDraw);
-    core.doAction();
-}
-
-events.prototype._action_input = function (data, x, y, prefix) {
-    this.__action_getInput(core.replaceText(data.text, prefix), false, function (value) {
-        value = parseInt(value) || 0; // 允许负整数
-        core.status.route.push("input:" + value);
-        core.setFlag("input", value);
-        core.doAction();
-    });
-}
-
-events.prototype._action_input2 = function (data, x, y, prefix) {
-    this.__action_getInput(core.replaceText(data.text, prefix), true, function (value) {
-        value = value || "";
-        core.status.route.push("input2:" + core.encodeBase64(value));
-        core.setFlag("input", value);
-        core.doAction();
-    });
-}
-
-events.prototype.__action_getInput = function (hint, isText, callback) {
-    var value, prefix = isText ? "input2:" : "input:";
-    if (core.isReplaying()) {
-        var action = core.status.replay.toReplay.shift();
-        try {
-            if (action.indexOf(prefix) != 0) {
-                console.warn("警告!当前需要一个 " + prefix + " 项,实际为 " + action);
-                core.status.replay.toReplay.unshift(action);
-                return callback(isText ? '' : 0);
-            }
-            if (isText) value = core.decodeBase64(action.substring(7));
-            else value = parseInt(action.substring(6));
-            callback(value);
-        }
-        catch (e) {
-            core.control._replay_error(action);
-            return;
-        }
-    }
-    else {
-        core.myprompt(core.replaceText(hint), null, callback);
-    }
-}
-
-events.prototype._action_if = function (data, x, y, prefix) {
-    if (core.calValue(data.condition, prefix))
-        core.events.insertAction(data["true"])
-    else
-        core.events.insertAction(data["false"])
-    core.doAction();
-}
-
-events.prototype._precompile_if = function (data) {
-    data.condition = core.replaceValue(data.condition);
-    data["true"] = this.precompile(data["true"]);
-    data["false"] = this.precompile(data["false"]);
-    return data;
-}
-
-events.prototype._action_switch = function (data, x, y, prefix) {
-    var key = core.calValue(data.condition, prefix)
-    var list = [];
-    for (var i = 0; i < data.caseList.length; i++) {
-        if (data.caseList[i]._disabled) continue;
-        var condition = data.caseList[i]["case"];
-        if (condition == "default" || core.calValue(condition, prefix) === key) {
-            core.push(list, data.caseList[i].action);
-            if (!data.caseList[i].nobreak)
-                break;
-        }
-    }
-    core.insertAction(list);
-    core.doAction();
-}
-
-events.prototype._precompile_switch = function (data) {
-    data.condition = core.replaceValue(data.condition);
-    for (var i = 0; i < data.caseList.length; i++) {
-        data.caseList[i]["case"] = core.replaceValue(data.caseList[i]["case"]);
-        data.caseList[i].action = this.precompile(data.caseList[i].action);
-    }
-    return data;
-}
-
-events.prototype._action_choices = function (data, x, y, prefix) {
-    data.choices = data.choices.filter(function (x) {
-        if (x._disabled) return false;
-        if (x.condition == null || x.condition == '') return true;
-        try { return core.calValue(x.condition, prefix); } catch (e) { return true; }
-    })
-    if (data.choices.length == 0) return this.doAction();
-    if (core.isReplaying()) {
-        var action = core.status.replay.toReplay.shift();
-        if (action.indexOf('choices:') == 0 && !(action == 'choices:none' && !data.timeout)) {
-            var index = action.substring(8);
-            if (!this.__action_choices_replaying(data, index)) {
-                core.control._replay_error(action);
-                return;
-            }
-        } else {
-            // 容错录像
-            if (main.replayChecking) {
-                // 录像验证系统中选最后一项
-                if (action != 'choices:none') core.status.replay.toReplay.unshift(action); // 首先归还刚才读出的下一步操作
-                core.events.__action_choices_replaying(data, -1);
-            } else {
-                // 正常游戏中弹窗选择
-                core.myprompt('录像回放出错!当前需要执行选择项但录像中未记录。\n如需修复请输入您要选的项(从0起),点击取消将不会修复。', 0, function (value) {
-                    if (value == null) {
-                        core.control._replay_error(action);
-                        return;
-                    }
-                    if (action != 'choices:none') core.status.replay.toReplay.unshift(action); // 首先归还刚才读出的下一步操作
-                    core.events.__action_choices_replaying(data, ((parseInt(value) || 0) + data.choices.length) % data.choices.length);
-                });
-            }
-        }
-    } else {
-        if (data.timeout) {
-            core.status.event.interval = setTimeout(function () {
-                core.status.route.push("choices:none");
-                core.setFlag('timeout', 0);
-                core.doAction();
-            }, data.timeout);
-        }
-        core.status.event.timeout = new Date().getTime() + (data.timeout || 0);
-    }
-    for (var i = 0; i < data.choices.length; i++) {
-        if (typeof data.choices[i] === 'string')
-            data.choices[i] = { "text": data.choices[i] };
-        data.choices[i].text = core.replaceText(data.choices[i].text, prefix);
-    }
-    core.ui.drawChoices(core.replaceText(data.text, prefix), data.choices, data.width);
-}
-
-events.prototype.__action_choices_replaying = function (data, index) {
-    var selection = index;
-    if (index != 'none') {
-        selection = parseInt(index);
-        if (isNaN(selection)) return false;
-        if (selection < 0) selection += data.choices.length;
-        if (selection < 0) return false;
-        if (selection % 100 > 50) selection += data.choices.length;
-        if (selection % 100 > data.choices.length) return false;
-        var timeout = Math.floor(selection / 100) || 0;
-        core.setFlag('timeout', timeout);
-        selection %= 100;
-    } else core.setFlag('timeout', 0);
-    core.status.event.selection = selection;
-    setTimeout(function () {
-        core.status.route.push("choices:" + index);
-        if (selection != 'none') {
-            // 检查
-            var choice = data.choices[selection];
-            if (choice.need != null && choice.need != '' && !core.calValue(choice.need)) {
-                // 无法选择此项
-                core.control._replay_error("无法选择项:" + index);
-                return;
-            } else {
-                core.insertAction(choice.action);
-            }
-        }
-        core.doAction();
-    }, core.status.replay.speed == 24 ? 1 : 750 / Math.max(1, core.status.replay.speed));
-    return true;
-}
-
-events.prototype._precompile_choices = function (data) {
-    if (!(data.choices instanceof Array)) return data;
-    for (var i = 0; i < data.choices.length; ++i) {
-        data.choices[i].condition = core.replaceValue(data.choices[i].condition);
-        data.choices[i].text = this.__precompile_text(data.choices[i].text);
-        data.choices[i].action = this.precompile(data.choices[i].action);
-    }
-    return data;
-}
-
-events.prototype._action_confirm = function (data, x, y, prefix) {
-    data.text = core.replaceText(data.text, prefix);
-    core.status.event.ui = { "text": data.text, "yes": data.yes, "no": data.no };
-    if (core.isReplaying()) {
-        var action = core.status.replay.toReplay.shift();
-        if (action.indexOf('choices:') == 0 && !(action == 'choices:none' && !data.timeout)) {
-            var index = action.substring(8);
-            if (index == 'none' || ((index = parseInt(index)) >= 0) && index % 100 < 2) {
-                this.__action_confirm_replaying(data, index);
-            } else {
-                core.control._replay_error(action);
-                return;
-            }
-        } else {
-            // 录像中未记录选了哪个,则选默认值,而不是直接报错
-            if (action != 'choices:none') core.status.replay.toReplay.unshift(action);
-            this.__action_confirm_replaying(data, data["default"] ? 0 : 1);
-        }
-    }
-    else {
-        core.status.event.selection = data["default"] ? 0 : 1;
-        if (data.timeout) {
-            core.status.event.interval = setTimeout(function () {
-                core.status.route.push("choices:none");
-                core.setFlag('timeout', 0);
-                core.doAction();
-            }, data.timeout);
-        }
-        core.status.event.timeout = new Date().getTime() + (data.timeout || 0);
-    }
-    core.ui.drawConfirmBox(data.text);
-}
-
-events.prototype.__action_confirm_replaying = function (data, index) {
-    if (index != 'none') {
-        var timeout = Math.floor(index / 100) || 0;
-        core.setFlag('timeout', timeout);
-        index %= 100;
-    } else core.setFlag('timeout', 0);
-    core.status.event.selection = index;
-    setTimeout(function () {
-        core.status.route.push("choices:" + index);
-        if (index != 'none') {
-            if (index == 0) core.insertAction(data.yes);
-            else core.insertAction(data.no);
-        }
-        core.doAction();
-    }, core.status.replay.speed == 24 ? 1 : 750 / Math.max(1, core.status.replay.speed));
-}
-
-events.prototype._precompile_confirm = function (data) {
-    data.yes = this.precompile(data.yes);
-    data.no = this.precompile(data.no);
-    return data;
-}
-
-events.prototype._action_for = function (data, x, y, prefix) {
-    // Only support temp:A
-    if (!/^temp:[A-Z]$/.test(data.name)) {
-        core.insertAction('循环遍历事件只支持临时变量!');
-        return core.doAction();
-    }
-    var from = core.calValue(data.from);
-    var to = core.calValue(data.to);
-    var step = core.calValue(data.step);
-    if (typeof from != 'number' || typeof to != 'number' || typeof step != 'number') {
-        core.insertAction('循环遍历事件要求【起始点】【终止点】【每步】仅能是数字!');
-        return core.doAction();
-    }
-    // 首次判定
-    if ((step > 0 && from > to) || (step < 0 && from < to)) {
-        core.doAction();
-        return;
-    }
-
-    var letter = data.name.substring(5);
-    core.setFlag('@temp@' + letter, from);
-    var toName = '@temp@for-to@' + letter;
-    var stepName = '@temp@for-step@' + letter;
-    core.setFlag(toName, data.to);
-    core.setFlag(stepName, data.step);
-    var condition = "(function () {" +
-        "var to = core.calValue(core.getFlag('" + toName + "'));" +
-        "var step = core.calValue(core.getFlag('" + stepName + "'));" +
-        "if (typeof step != 'number' || typeof to != 'number') return false;" +
-        "if (step == 0) return true;" +
-        "var currentValue = core.getFlag('@temp@" + letter + "');" +
-        "currentValue += step;" +
-        "core.setFlag('@temp@" + letter + "', currentValue);" +
-        "if (step > 0) { return currentValue <= to; }" +
-        "else { return currentValue >= to; }" +
-        "})()";
-    return this._action_dowhile({ "condition": condition, "data": data.data }, x, y, prefix);
-}
-
-events.prototype._precompile_for = function (data) {
-    data.from = core.replaceValue(data.from);
-    data.to = core.replaceValue(data.to);
-    data.step = core.replaceValue(data.step);
-    data.data = this.precompile(data.data);
-    return data;
-}
-
-events.prototype._action_forEach = function (data, x, y, prefix) {
-    // Only support temp:A
-    if (!/^temp:[A-Z]$/.test(data.name)) {
-        core.insertAction(['循环遍历事件只支持临时变量!']);
-        return core.doAction();
-    }
-    var listName = '@temp@forEach@' + data.name.substring(5);
-    core.setFlag(listName, core.clone(data.list));
-    var condition = "(function () {" +
-        "var list = core.getFlag('" + listName + "', []);" +
-        "if (list.length == 0) return false;" +
-        "core.setFlag('@temp@'+'" + data.name.substring(5) + "', list.shift());" +
-        "return true;" +
-        "})()";
-    return this._action_while({ "condition": condition, "data": data.data }, x, y, prefix);
-}
-
-events.prototype._precompile_forEach = function (data) {
-    data.data = this.precompile(data.data);
-    return data;
-}
-
-events.prototype._action_while = function (data, x, y, prefix) {
-    if (core.calValue(data.condition, prefix)) {
-        var list = core.clone(data.data);
-        if (!(list instanceof Array)) list = [list];
-        list.push({ "type": "_label" });
-        core.unshift(core.status.event.data.list,
-            { "todo": list, "total": core.clone(list), "condition": data.condition }
-        );
-    }
-    core.doAction();
-}
-
-events.prototype._precompile_while = function (data) {
-    data.condition = core.replaceValue(data.condition);
-    data.data = this.precompile(data.data);
-    return data;
-}
-
-events.prototype._action_dowhile = function (data, x, y, prefix) {
-    var list = core.clone(data.data);
-    if (!(list instanceof Array)) list = [list];
-    list.push({ "type": "_label" });
-    core.unshift(core.status.event.data.list,
-        { "todo": list, "total": core.clone(list), "condition": data.condition }
+  // 考虑到可能覆盖楼传事件的问题,这里不对fly进行检查。
+  if (data.id != "book" && core.canUseItem(data.id)) {
+    core.useItem(data.id, true, core.doAction);
+  } else {
+    core.playSound("操作失败");
+    core.drawTip(
+      "当前无法使用" + ((core.material.items[data.id] || {}).name || "未知道具")
     );
     core.doAction();
-}
+  }
+};
 
-events.prototype._precompile_dowhile = function (data) {
-    data.condition = core.replaceValue(data.condition);
-    data.data = this.precompile(data.data);
-    return data;
-}
+events.prototype._action_loadEquip = function (data, x, y, prefix) {
+  core.loadEquip(data.id);
+  core.doAction();
+};
 
-events.prototype._action_break = function (data, x, y, prefix) {
-    var n = data.n || 1;
-    while (n--) {
-        if (core.status.event.data.list.length > 1)
-            core.status.event.data.list.shift();
+events.prototype._action_unloadEquip = function (data, x, y, prefix) {
+  core.unloadEquip(data.pos);
+  core.doAction();
+};
+
+events.prototype._action_openShop = function (data, x, y, prefix) {
+  core.setShopVisited(data.id, true);
+  if (data.open) core.openShop(data.id, true);
+  core.doAction();
+};
+
+events.prototype._action_disableShop = function (data, x, y, prefix) {
+  core.setShopVisited(data.id, false);
+  core.doAction();
+};
+
+events.prototype._action_battle = function (data, x, y, prefix) {
+  if (data.id) {
+    this.battle(data.id, null, null, true, core.doAction);
+  } else {
+    if (data.floorId != core.status.floorId) {
+      core.doAction();
+      return;
     }
-    core.doAction();
-}
+    var loc = this.__action_getLoc(data.loc, x, y, prefix);
+    this.battle(null, loc[0], loc[1], true, core.doAction);
+  }
+};
 
-events.prototype._action_continue = function (data, x, y, prefix) {
-    var n = data.n || 1;
-    while (n-- > 1) {
-        if (core.status.event.data.list.length > 1)
-            core.status.event.data.list.shift();
-    }
-    if (core.status.event.data.list.length > 1) {
-        if (core.calValue(core.status.event.data.list[0].condition, prefix)) {
-            core.status.event.data.list[0].todo = core.clone(core.status.event.data.list[0].total);
-        }
-        else {
-            core.status.event.data.list.shift();
+events.prototype._action_trigger = function (data, x, y, prefix) {
+  var loc = this.__action_getLoc(data.loc, x, y, prefix);
+  this._trigger_inAction(loc[0], loc[1]);
+};
+
+events.prototype._action_insert = function (data, x, y, prefix) {
+  if (data.name) {
+    // 公共事件
+    core.insertCommonEvent(data.name, data.args);
+  } else {
+    // 设置参数
+    if (data.args instanceof Array) {
+      for (var i = 0; i < data.args.length; ++i) {
+        try {
+          if (data.args[i] != null) core.setFlag("arg" + (i + 1), data.args[i]);
+        } catch (ee) {
+          console.error(ee);
         }
+      }
     }
+    var loc = this.__action_getLoc(data.loc, x, y, prefix);
+    core.setFlag("arg0", loc);
+    var floorId = data.floorId;
+    var which = data.which || "events";
+    var event = (core.floors[floorId][which] || [])[loc[0] + "," + loc[1]];
+    if (event) this.insertAction(event.data || event);
+  }
+  core.doAction();
+};
+
+events.prototype._action_playBgm = function (data, x, y, prefix) {
+  core.playBgm(data.name, data.startTime || 0);
+  core.setFlag("__bgm__", data.keep ? data.name : null);
+  core.doAction();
+};
+
+events.prototype._action_pauseBgm = function (data, x, y, prefix) {
+  core.pauseBgm();
+  core.doAction();
+};
+
+events.prototype._action_resumeBgm = function (data, x, y, prefix) {
+  core.resumeBgm(data.resume);
+  core.doAction();
+};
+
+events.prototype._action_loadBgm = function (data, x, y, prefix) {
+  core.loadBgm(data.name);
+  core.doAction();
+};
+
+events.prototype._action_freeBgm = function (data, x, y, prefix) {
+  core.freeBgm(data.name);
+  core.doAction();
+};
+
+events.prototype._action_playSound = function (data, x, y, prefix) {
+  if (data.stop) core.stopSound();
+  if (data.sync) {
+    core.playSound(data.name, data.pitch, core.doAction);
+  } else {
+    core.playSound(data.name, data.pitch);
     core.doAction();
-}
+  }
+};
 
-events.prototype._action_win = function (data, x, y, prefix) {
-    this.win(core.replaceText(data.reason, prefix), data.norank, data.noexit);
-}
+events.prototype._action_stopSound = function (data, x, y, prefix) {
+  core.stopSound();
+  core.doAction();
+};
 
-events.prototype._action_lose = function (data, x, y, prefix) {
-    this.lose(core.replaceText(data.reason, prefix));
-}
+events.prototype._action_setVolume = function (data, x, y, prefix) {
+  data.value = core.clamp(parseInt(data.value) / 100, 0, 1);
+  core.setFlag("__volume__", data.value);
+  this.__action_doAsyncFunc(
+    data.async,
+    core.setVolume,
+    data.value,
+    data.time || 0
+  );
+};
 
-events.prototype._action_restart = function (data, x, y, prefix) {
-    core.restart();
-}
+events.prototype._action_setBgmSpeed = function (data, x, y, prefix) {
+  core.setBgmSpeed(data.value, data.pitch || false);
+  core.doAction();
+};
 
-events.prototype._action_function = function (data, x, y, prefix) {
-    var func = data["function"];
-    try {
-        if (typeof func == "string" && func.indexOf("function") == 0) {
-            eval('(' + func + ')()');
-        }
-    } catch (e) {
-        console.error(e);
-    }
-    if (!data.async)
-        core.doAction();
-}
-
-events.prototype._action_update = function (data, x, y, prefix) {
-    core.updateStatusBar(data.doNotCheckAutoEvents, true);
-    core.doAction();
-}
-
-events.prototype._action_showStatusBar = function (data, x, y, prefix) {
-    core.showStatusBar();
-    core.doAction();
-}
-
-events.prototype._action_hideStatusBar = function (data, x, y, prefix) {
-    core.hideStatusBar(data.toolbox);
-    core.doAction();
-}
-
-events.prototype._action_showHero = function (data, x, y, prefix) {
-    data.opacity = 1;
-    return this._action_setHeroOpacity(data, x, y, prefix);
-}
-
-events.prototype._action_hideHero = function (data, x, y, prefix) {
-    data.opacity = 0;
-    return this._action_setHeroOpacity(data, x, y, prefix);
-}
-
-events.prototype._action_setHeroOpacity = function (data, x, y, prefix) {
-    data.time = data.time || 0;
-    if (data.opacity == null) data.opacity = 1;
-    if (data.time > 0) {
-        this.__action_doAsyncFunc(data.async, core.setHeroOpacity, data.opacity, data.moveMode, data.time);
+events.prototype._action_setValue = function (data, x, y, prefix) {
+  this.setValue(data.name, data.operator, data.value, prefix);
+  if (!data.norefresh) {
+    if (core.status.hero.hp <= 0) {
+      core.status.hero.hp = 0;
+      core.updateStatusBar();
+      core.events.lose();
     } else {
-        core.setFlag('__heroOpacity__', data.opacity);
-        core.drawHero();
-        core.doAction();
+      core.updateStatusBar();
     }
-}
+  }
+  core.doAction();
+};
 
-events.prototype._action_vibrate = function (data, x, y, prefix) {
-    this.__action_doAsyncFunc(data.async, core.vibrate, data.direction, data.time, data.speed, data.power);
-}
+events.prototype._action_addValue = function (data, x, y, prefix) {
+  data.operator = "+=";
+  this._action_setValue(data, x, y, prefix);
+};
 
-events.prototype._action_sleep = function (data, x, y, prefix) {
-    core.timeout.sleepTimeout = setTimeout(function () {
-        core.timeout.sleepTimeout = null;
-        core.doAction();
-    }, core.isReplaying() ? Math.min(data.time, 20) : data.time);
-}
+events.prototype._action_setEnemy = function (data, x, y, prefix) {
+  this.setEnemy(
+    data.id,
+    data.name,
+    data.value,
+    data.operator,
+    prefix,
+    data.norefresh
+  );
+  core.doAction();
+};
 
-events.prototype._action_wait = function (data, x, y, prefix) {
-    if (core.isReplaying()) {
-        var code = core.status.replay.toReplay.shift();
-        if (code.indexOf("input:") == 0 && !(code == "input:none" && !data.timeout)) {
-            if (code == "input:none") {
-                core.status.route.push("input:none");
-                core.setFlag("type", -1);
-                core.setFlag("timeout", 0);
-                this.__action_wait_afterGet(data);
-                return core.doAction();
-            } else {
-                var value = parseInt(code.substring(6));
-                core.status.route.push("input:" + value);
-                this.__action_wait_getValue(value);
-                if (this.__action_wait_afterGet(data) || !data.forceChild) return core.doAction();
-            }
-        }
-        core.control._replay_error(code);
+events.prototype._action_setEnemyOnPoint = function (data, x, y, prefix) {
+  var loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  loc.forEach(function (one) {
+    core.setEnemyOnPoint(
+      one[0],
+      one[1],
+      data.floorId,
+      data.name,
+      data.value,
+      data.operator,
+      prefix,
+      data.norefresh
+    );
+  });
+  core.doAction();
+};
+
+events.prototype._action_resetEnemyOnPoint = function (data, x, y, prefix) {
+  var loc = this.__action_getLoc2D(data.loc, x, y, prefix);
+  loc.forEach(function (one) {
+    core.resetEnemyOnPoint(one[0], one[1], data.floorId, data.norefresh);
+  });
+  core.doAction();
+};
+
+events.prototype._precompile_moveEnemyOnPoint = function (data) {
+  data.from = this.__precompile_array(data.from);
+  data.to = this.__precompile_array(data.to);
+  data.dxy = this.__precompile_array(data.dxy);
+  return data;
+};
+
+events.prototype._action_moveEnemyOnPoint = function (data, x, y, prefix) {
+  var from = this.__action_getLoc(data.from, x, y, prefix),
+    to;
+  if (data.dxy) {
+    to = [
+      from[0] + (core.calValue(data.dxy[0], prefix) || 0),
+      from[1] + (core.calValue(data.dxy[1], prefix) || 0),
+    ];
+  } else {
+    to = this.__action_getLoc(data.to, x, y, prefix);
+  }
+  this.moveEnemyOnPoint(
+    from[0],
+    from[1],
+    to[0],
+    to[1],
+    data.floorId,
+    data.norefresh
+  );
+  core.doAction();
+};
+
+events.prototype._action_setEquip = function (data, x, y, prefix) {
+  core.setEquip(
+    data.id,
+    data.valueType,
+    data.name,
+    data.value,
+    data.operator,
+    prefix
+  );
+  core.doAction();
+};
+
+events.prototype._action_setFloor = function (data, x, y, prefix) {
+  this.setFloorInfo(data.name, data.value, data.floorId, prefix);
+  core.doAction();
+};
+
+events.prototype._action_setGlobalAttribute = function (data, x, y, prefix) {
+  this.setGlobalAttribute(data.name, data.value);
+  core.doAction();
+};
+
+events.prototype._action_setGlobalValue = function (data, x, y, prefix) {
+  core.values[data.name] = data.value;
+  core.doAction();
+};
+
+events.prototype._action_setGlobalFlag = function (data, x, y, prefix) {
+  this.setGlobalFlag(data.name, data.value);
+  core.doAction();
+};
+
+events.prototype._action_setNameMap = function (data, x, y, floorId) {
+  this.setNameMap(data.name, data.value);
+  core.doAction();
+};
+
+events.prototype._action_setHeroIcon = function (data, x, y, prefix) {
+  this.setHeroIcon(data.name, data.noDraw);
+  core.doAction();
+};
+
+events.prototype._action_input = function (data, x, y, prefix) {
+  this.__action_getInput(
+    core.replaceText(data.text, prefix),
+    false,
+    function (value) {
+      value = parseInt(value) || 0; // 允许负整数
+      core.status.route.push("input:" + value);
+      core.setFlag("input", value);
+      core.doAction();
+    }
+  );
+};
+
+events.prototype._action_input2 = function (data, x, y, prefix) {
+  this.__action_getInput(
+    core.replaceText(data.text, prefix),
+    true,
+    function (value) {
+      value = value || "";
+      core.status.route.push("input2:" + core.encodeBase64(value));
+      core.setFlag("input", value);
+      core.doAction();
+    }
+  );
+};
+
+events.prototype.__action_getInput = function (hint, isText, callback) {
+  var value,
+    prefix = isText ? "input2:" : "input:";
+  if (core.isReplaying()) {
+    var action = core.status.replay.toReplay.shift();
+    try {
+      if (action.indexOf(prefix) != 0) {
+        console.warn("警告!当前需要一个 " + prefix + " 项,实际为 " + action);
+        core.status.replay.toReplay.unshift(action);
+        return callback(isText ? "" : 0);
+      }
+      if (isText) value = core.decodeBase64(action.substring(7));
+      else value = parseInt(action.substring(6));
+      callback(value);
+    } catch (e) {
+      core.control._replay_error(action);
+      return;
+    }
+  } else {
+    core.myprompt(core.replaceText(hint), null, callback);
+  }
+};
+
+events.prototype._action_if = function (data, x, y, prefix) {
+  if (core.calValue(data.condition, prefix))
+    core.events.insertAction(data["true"]);
+  else core.events.insertAction(data["false"]);
+  core.doAction();
+};
+
+events.prototype._precompile_if = function (data) {
+  data.condition = core.replaceValue(data.condition);
+  data["true"] = this.precompile(data["true"]);
+  data["false"] = this.precompile(data["false"]);
+  return data;
+};
+
+events.prototype._action_switch = function (data, x, y, prefix) {
+  var key = core.calValue(data.condition, prefix);
+  var list = [];
+  for (var i = 0; i < data.caseList.length; i++) {
+    if (data.caseList[i]._disabled) continue;
+    var condition = data.caseList[i]["case"];
+    if (condition == "default" || core.calValue(condition, prefix) === key) {
+      core.push(list, data.caseList[i].action);
+      if (!data.caseList[i].nobreak) break;
+    }
+  }
+  core.insertAction(list);
+  core.doAction();
+};
+
+events.prototype._precompile_switch = function (data) {
+  data.condition = core.replaceValue(data.condition);
+  for (var i = 0; i < data.caseList.length; i++) {
+    data.caseList[i]["case"] = core.replaceValue(data.caseList[i]["case"]);
+    data.caseList[i].action = this.precompile(data.caseList[i].action);
+  }
+  return data;
+};
+
+events.prototype._action_choices = function (data, x, y, prefix) {
+  data.choices = data.choices.filter(function (x) {
+    if (x._disabled) return false;
+    if (x.condition == null || x.condition == "") return true;
+    try {
+      return core.calValue(x.condition, prefix);
+    } catch (e) {
+      return true;
+    }
+  });
+  if (data.choices.length == 0) return this.doAction();
+  if (core.isReplaying()) {
+    var action = core.status.replay.toReplay.shift();
+    if (
+      action.indexOf("choices:") == 0 &&
+      !(action == "choices:none" && !data.timeout)
+    ) {
+      var index = action.substring(8);
+      if (!this.__action_choices_replaying(data, index)) {
+        core.control._replay_error(action);
         return;
-    } else if (data.timeout) {
-        core.status.event.interval = setTimeout(function () {
-            core.status.route.push("input:none");
-            core.setFlag("type", -1);
-            core.setFlag("timeout", 0);
-            core.events.__action_wait_afterGet(data);
-            core.doAction();
-        }, data.timeout);
+      }
+    } else {
+      // 容错录像
+      if (main.replayChecking) {
+        // 录像验证系统中选最后一项
+        if (action != "choices:none")
+          core.status.replay.toReplay.unshift(action); // 首先归还刚才读出的下一步操作
+        core.events.__action_choices_replaying(data, -1);
+      } else {
+        // 正常游戏中弹窗选择
+        core.myprompt(
+          "录像回放出错!当前需要执行选择项但录像中未记录。\n如需修复请输入您要选的项(从0起),点击取消将不会修复。",
+          0,
+          function (value) {
+            if (value == null) {
+              core.control._replay_error(action);
+              return;
+            }
+            if (action != "choices:none")
+              core.status.replay.toReplay.unshift(action); // 首先归还刚才读出的下一步操作
+            core.events.__action_choices_replaying(
+              data,
+              ((parseInt(value) || 0) + data.choices.length) %
+                data.choices.length
+            );
+          }
+        );
+      }
+    }
+  } else {
+    if (data.timeout) {
+      core.status.event.interval = setTimeout(function () {
+        core.status.route.push("choices:none");
+        core.setFlag("timeout", 0);
+        core.doAction();
+      }, data.timeout);
     }
     core.status.event.timeout = new Date().getTime() + (data.timeout || 0);
-}
+  }
+  for (var i = 0; i < data.choices.length; i++) {
+    if (typeof data.choices[i] === "string")
+      data.choices[i] = { text: data.choices[i] };
+    data.choices[i].text = core.replaceText(data.choices[i].text, prefix);
+  }
+  core.ui.drawChoices(
+    core.replaceText(data.text, prefix),
+    data.choices,
+    data.width
+  );
+};
+
+events.prototype.__action_choices_replaying = function (data, index) {
+  var selection = index;
+  if (index != "none") {
+    selection = parseInt(index);
+    if (isNaN(selection)) return false;
+    if (selection < 0) selection += data.choices.length;
+    if (selection < 0) return false;
+    if (selection % 100 > 50) selection += data.choices.length;
+    if (selection % 100 > data.choices.length) return false;
+    var timeout = Math.floor(selection / 100) || 0;
+    core.setFlag("timeout", timeout);
+    selection %= 100;
+  } else core.setFlag("timeout", 0);
+  core.status.event.selection = selection;
+  setTimeout(
+    function () {
+      core.status.route.push("choices:" + index);
+      if (selection != "none") {
+        // 检查
+        var choice = data.choices[selection];
+        if (
+          choice.need != null &&
+          choice.need != "" &&
+          !core.calValue(choice.need)
+        ) {
+          // 无法选择此项
+          core.control._replay_error("无法选择项:" + index);
+          return;
+        } else {
+          core.insertAction(choice.action);
+        }
+      }
+      core.doAction();
+    },
+    core.status.replay.speed == 24
+      ? 1
+      : 750 / Math.max(1, core.status.replay.speed)
+  );
+  return true;
+};
+
+events.prototype._precompile_choices = function (data) {
+  if (!(data.choices instanceof Array)) return data;
+  for (var i = 0; i < data.choices.length; ++i) {
+    data.choices[i].condition = core.replaceValue(data.choices[i].condition);
+    data.choices[i].text = this.__precompile_text(data.choices[i].text);
+    data.choices[i].action = this.precompile(data.choices[i].action);
+  }
+  return data;
+};
+
+events.prototype._action_confirm = function (data, x, y, prefix) {
+  data.text = core.replaceText(data.text, prefix);
+  core.status.event.ui = { text: data.text, yes: data.yes, no: data.no };
+  if (core.isReplaying()) {
+    var action = core.status.replay.toReplay.shift();
+    if (
+      action.indexOf("choices:") == 0 &&
+      !(action == "choices:none" && !data.timeout)
+    ) {
+      var index = action.substring(8);
+      if (
+        index == "none" ||
+        ((index = parseInt(index)) >= 0 && index % 100 < 2)
+      ) {
+        this.__action_confirm_replaying(data, index);
+      } else {
+        core.control._replay_error(action);
+        return;
+      }
+    } else {
+      // 录像中未记录选了哪个,则选默认值,而不是直接报错
+      if (action != "choices:none") core.status.replay.toReplay.unshift(action);
+      this.__action_confirm_replaying(data, data["default"] ? 0 : 1);
+    }
+  } else {
+    core.status.event.selection = data["default"] ? 0 : 1;
+    if (data.timeout) {
+      core.status.event.interval = setTimeout(function () {
+        core.status.route.push("choices:none");
+        core.setFlag("timeout", 0);
+        core.doAction();
+      }, data.timeout);
+    }
+    core.status.event.timeout = new Date().getTime() + (data.timeout || 0);
+  }
+  core.ui.drawConfirmBox(data.text);
+};
+
+events.prototype.__action_confirm_replaying = function (data, index) {
+  if (index != "none") {
+    var timeout = Math.floor(index / 100) || 0;
+    core.setFlag("timeout", timeout);
+    index %= 100;
+  } else core.setFlag("timeout", 0);
+  core.status.event.selection = index;
+  setTimeout(
+    function () {
+      core.status.route.push("choices:" + index);
+      if (index != "none") {
+        if (index == 0) core.insertAction(data.yes);
+        else core.insertAction(data.no);
+      }
+      core.doAction();
+    },
+    core.status.replay.speed == 24
+      ? 1
+      : 750 / Math.max(1, core.status.replay.speed)
+  );
+};
+
+events.prototype._precompile_confirm = function (data) {
+  data.yes = this.precompile(data.yes);
+  data.no = this.precompile(data.no);
+  return data;
+};
+
+events.prototype._action_for = function (data, x, y, prefix) {
+  // Only support temp:A
+  if (!/^temp:[A-Z]$/.test(data.name)) {
+    core.insertAction("循环遍历事件只支持临时变量!");
+    return core.doAction();
+  }
+  var from = core.calValue(data.from);
+  var to = core.calValue(data.to);
+  var step = core.calValue(data.step);
+  if (
+    typeof from != "number" ||
+    typeof to != "number" ||
+    typeof step != "number"
+  ) {
+    core.insertAction(
+      "循环遍历事件要求【起始点】【终止点】【每步】仅能是数字!"
+    );
+    return core.doAction();
+  }
+  // 首次判定
+  if ((step > 0 && from > to) || (step < 0 && from < to)) {
+    core.doAction();
+    return;
+  }
+
+  var letter = data.name.substring(5);
+  core.setFlag("@temp@" + letter, from);
+  var toName = "@temp@for-to@" + letter;
+  var stepName = "@temp@for-step@" + letter;
+  core.setFlag(toName, data.to);
+  core.setFlag(stepName, data.step);
+  var condition =
+    "(function () {" +
+    "var to = core.calValue(core.getFlag('" +
+    toName +
+    "'));" +
+    "var step = core.calValue(core.getFlag('" +
+    stepName +
+    "'));" +
+    "if (typeof step != 'number' || typeof to != 'number') return false;" +
+    "if (step == 0) return true;" +
+    "var currentValue = core.getFlag('@temp@" +
+    letter +
+    "');" +
+    "currentValue += step;" +
+    "core.setFlag('@temp@" +
+    letter +
+    "', currentValue);" +
+    "if (step > 0) { return currentValue <= to; }" +
+    "else { return currentValue >= to; }" +
+    "})()";
+  return this._action_dowhile(
+    { condition: condition, data: data.data },
+    x,
+    y,
+    prefix
+  );
+};
+
+events.prototype._precompile_for = function (data) {
+  data.from = core.replaceValue(data.from);
+  data.to = core.replaceValue(data.to);
+  data.step = core.replaceValue(data.step);
+  data.data = this.precompile(data.data);
+  return data;
+};
+
+events.prototype._action_forEach = function (data, x, y, prefix) {
+  // Only support temp:A
+  if (!/^temp:[A-Z]$/.test(data.name)) {
+    core.insertAction(["循环遍历事件只支持临时变量!"]);
+    return core.doAction();
+  }
+  var listName = "@temp@forEach@" + data.name.substring(5);
+  core.setFlag(listName, core.clone(data.list));
+  var condition =
+    "(function () {" +
+    "var list = core.getFlag('" +
+    listName +
+    "', []);" +
+    "if (list.length == 0) return false;" +
+    "core.setFlag('@temp@'+'" +
+    data.name.substring(5) +
+    "', list.shift());" +
+    "return true;" +
+    "})()";
+  return this._action_while(
+    { condition: condition, data: data.data },
+    x,
+    y,
+    prefix
+  );
+};
+
+events.prototype._precompile_forEach = function (data) {
+  data.data = this.precompile(data.data);
+  return data;
+};
+
+events.prototype._action_while = function (data, x, y, prefix) {
+  if (core.calValue(data.condition, prefix)) {
+    var list = core.clone(data.data);
+    if (!(list instanceof Array)) list = [list];
+    list.push({ type: "_label" });
+    core.unshift(core.status.event.data.list, {
+      todo: list,
+      total: core.clone(list),
+      condition: data.condition,
+    });
+  }
+  core.doAction();
+};
+
+events.prototype._precompile_while = function (data) {
+  data.condition = core.replaceValue(data.condition);
+  data.data = this.precompile(data.data);
+  return data;
+};
+
+events.prototype._action_dowhile = function (data, x, y, prefix) {
+  var list = core.clone(data.data);
+  if (!(list instanceof Array)) list = [list];
+  list.push({ type: "_label" });
+  core.unshift(core.status.event.data.list, {
+    todo: list,
+    total: core.clone(list),
+    condition: data.condition,
+  });
+  core.doAction();
+};
+
+events.prototype._precompile_dowhile = function (data) {
+  data.condition = core.replaceValue(data.condition);
+  data.data = this.precompile(data.data);
+  return data;
+};
+
+events.prototype._action_break = function (data, x, y, prefix) {
+  var n = data.n || 1;
+  while (n--) {
+    if (core.status.event.data.list.length > 1)
+      core.status.event.data.list.shift();
+  }
+  core.doAction();
+};
+
+events.prototype._action_continue = function (data, x, y, prefix) {
+  var n = data.n || 1;
+  while (n-- > 1) {
+    if (core.status.event.data.list.length > 1)
+      core.status.event.data.list.shift();
+  }
+  if (core.status.event.data.list.length > 1) {
+    if (core.calValue(core.status.event.data.list[0].condition, prefix)) {
+      core.status.event.data.list[0].todo = core.clone(
+        core.status.event.data.list[0].total
+      );
+    } else {
+      core.status.event.data.list.shift();
+    }
+  }
+  core.doAction();
+};
+
+events.prototype._action_win = function (data, x, y, prefix) {
+  this.win(core.replaceText(data.reason, prefix), data.norank, data.noexit);
+};
+
+events.prototype._action_lose = function (data, x, y, prefix) {
+  this.lose(core.replaceText(data.reason, prefix));
+};
+
+events.prototype._action_restart = function (data, x, y, prefix) {
+  core.restart();
+};
+
+events.prototype._action_function = function (data, x, y, prefix) {
+  var func = data["function"];
+  try {
+    if (typeof func == "string" && func.indexOf("function") == 0) {
+      eval("(" + func + ")()");
+    }
+  } catch (e) {
+    console.error(e);
+  }
+  if (!data.async) core.doAction();
+};
+
+events.prototype._action_update = function (data, x, y, prefix) {
+  core.updateStatusBar(data.doNotCheckAutoEvents, true);
+  core.doAction();
+};
+
+events.prototype._action_showStatusBar = function (data, x, y, prefix) {
+  core.showStatusBar();
+  core.doAction();
+};
+
+events.prototype._action_hideStatusBar = function (data, x, y, prefix) {
+  core.hideStatusBar(data.toolbox);
+  core.doAction();
+};
+
+events.prototype._action_showHero = function (data, x, y, prefix) {
+  data.opacity = 1;
+  return this._action_setHeroOpacity(data, x, y, prefix);
+};
+
+events.prototype._action_hideHero = function (data, x, y, prefix) {
+  data.opacity = 0;
+  return this._action_setHeroOpacity(data, x, y, prefix);
+};
+
+events.prototype._action_setHeroOpacity = function (data, x, y, prefix) {
+  data.time = data.time || 0;
+  if (data.opacity == null) data.opacity = 1;
+  if (data.time > 0) {
+    this.__action_doAsyncFunc(
+      data.async,
+      core.setHeroOpacity,
+      data.opacity,
+      data.moveMode,
+      data.time
+    );
+  } else {
+    core.setFlag("__heroOpacity__", data.opacity);
+    core.drawHero();
+    core.doAction();
+  }
+};
+
+events.prototype._action_vibrate = function (data, x, y, prefix) {
+  this.__action_doAsyncFunc(
+    data.async,
+    core.vibrate,
+    data.direction,
+    data.time,
+    data.speed,
+    data.power
+  );
+};
+
+events.prototype._action_sleep = function (data, x, y, prefix) {
+  core.timeout.sleepTimeout = setTimeout(
+    function () {
+      core.timeout.sleepTimeout = null;
+      core.doAction();
+    },
+    core.isReplaying() ? Math.min(data.time, 20) : data.time
+  );
+};
+
+events.prototype._action_wait = function (data, x, y, prefix) {
+  if (core.isReplaying()) {
+    var code = core.status.replay.toReplay.shift();
+    if (
+      code.indexOf("input:") == 0 &&
+      !(code == "input:none" && !data.timeout)
+    ) {
+      if (code == "input:none") {
+        core.status.route.push("input:none");
+        core.setFlag("type", -1);
+        core.setFlag("timeout", 0);
+        this.__action_wait_afterGet(data);
+        return core.doAction();
+      } else {
+        var value = parseInt(code.substring(6));
+        core.status.route.push("input:" + value);
+        this.__action_wait_getValue(value);
+        if (this.__action_wait_afterGet(data) || !data.forceChild)
+          return core.doAction();
+      }
+    }
+    core.control._replay_error(code);
+    return;
+  } else if (data.timeout) {
+    core.status.event.interval = setTimeout(function () {
+      core.status.route.push("input:none");
+      core.setFlag("type", -1);
+      core.setFlag("timeout", 0);
+      core.events.__action_wait_afterGet(data);
+      core.doAction();
+    }, data.timeout);
+  }
+  core.status.event.timeout = new Date().getTime() + (data.timeout || 0);
+};
 
 events.prototype.__action_wait_getValue = function (value) {
-    core.setFlag("timeout", Math.floor(value / 1e8) || 0);
-    value %= 1e8;
-    if (value >= 1000000) {
-        core.setFlag('type', 1);
-        var px = parseInt((value - 1000000) / 1000), py = value % 1000;
-        core.setFlag('px', px);
-        core.setFlag('py', py);
-        core.setFlag('x', parseInt(px / 32));
-        core.setFlag('y', parseInt(py / 32));
-    }
-    else if (value >= 10000) {
-        core.setFlag('type', 1);
-        var x = parseInt((value - 10000) / 100), y = value % 100;
-        core.setFlag('px', 32 * x + 16);
-        core.setFlag('py', 32 * y + 16);
-        core.setFlag('x', x);
-        core.setFlag('y', y);
-    }
-    else if (value > 0) {
-        core.setFlag('type', 0);
-        core.setFlag('keycode', value);
-    }
-}
+  core.setFlag("timeout", Math.floor(value / 1e8) || 0);
+  value %= 1e8;
+  if (value >= 1000000) {
+    core.setFlag("type", 1);
+    var px = parseInt((value - 1000000) / 1000),
+      py = value % 1000;
+    core.setFlag("px", px);
+    core.setFlag("py", py);
+    core.setFlag("x", parseInt(px / 32));
+    core.setFlag("y", parseInt(py / 32));
+  } else if (value >= 10000) {
+    core.setFlag("type", 1);
+    var x = parseInt((value - 10000) / 100),
+      y = value % 100;
+    core.setFlag("px", 32 * x + 16);
+    core.setFlag("py", 32 * y + 16);
+    core.setFlag("x", x);
+    core.setFlag("y", y);
+  } else if (value > 0) {
+    core.setFlag("type", 0);
+    core.setFlag("keycode", value);
+  }
+};
 
 events.prototype.__action_wait_afterGet = function (data) {
-    if (!data.data) return false;
-    var todo = [];
-    var stop = false;
-    var found = false;
-    data.data.forEach(function (one) {
-        if (one._disabled || stop) return;
-        if (one["case"] == "keyboard" && core.getFlag("type") == 0) {
-            (one.keycode + "").split(",").forEach(function (keycode) {
-                if (core.getFlag("keycode", 0) == keycode) {
-                    found = true;
-                    core.push(todo, one.action);
-                    if (one["break"]) stop = true;
-                }
-            });
+  if (!data.data) return false;
+  var todo = [];
+  var stop = false;
+  var found = false;
+  data.data.forEach(function (one) {
+    if (one._disabled || stop) return;
+    if (one["case"] == "keyboard" && core.getFlag("type") == 0) {
+      (one.keycode + "").split(",").forEach(function (keycode) {
+        if (core.getFlag("keycode", 0) == keycode) {
+          found = true;
+          core.push(todo, one.action);
+          if (one["break"]) stop = true;
         }
-        if (one["case"] == "mouse" && one.px instanceof Array
-            && one.py instanceof Array && core.getFlag("type") == 1) {
-            var pxmin = core.calValue(one.px[0]);
-            var pxmax = core.calValue(one.px[1]);
-            var pymin = core.calValue(one.py[0]);
-            var pymax = core.calValue(one.py[1]);
-            var px = core.getFlag("px", 0), py = core.getFlag("py", 0);
-            if (px >= pxmin && px <= pxmax && py >= pymin && py <= pymax) {
-                found = true;
-                core.push(todo, one.action);
-                if (one["break"]) stop = true;
-            }
-        }
-        if (one["case"] == "condition") {
-            var condition = false;
-            try { condition = core.calValue(one.condition); } catch (e) { }
-            if (condition) {
-                found = true;
-                core.push(todo, one.action);
-                if (one["break"]) stop = true;
-            }
-        }
-        if (one["case"] == "timeout" && core.getFlag("type") == -1) {
-            found = true;
-            core.push(todo, one.action);
-            if (one["break"]) stop = true;
-        }
-    })
-    if (found) {
-        core.insertAction(todo);
-        return true;
+      });
     }
-    return false;
-}
+    if (
+      one["case"] == "mouse" &&
+      one.px instanceof Array &&
+      one.py instanceof Array &&
+      core.getFlag("type") == 1
+    ) {
+      var pxmin = core.calValue(one.px[0]);
+      var pxmax = core.calValue(one.px[1]);
+      var pymin = core.calValue(one.py[0]);
+      var pymax = core.calValue(one.py[1]);
+      var px = core.getFlag("px", 0),
+        py = core.getFlag("py", 0);
+      if (px >= pxmin && px <= pxmax && py >= pymin && py <= pymax) {
+        found = true;
+        core.push(todo, one.action);
+        if (one["break"]) stop = true;
+      }
+    }
+    if (one["case"] == "condition") {
+      var condition = false;
+      try {
+        condition = core.calValue(one.condition);
+      } catch (e) {}
+      if (condition) {
+        found = true;
+        core.push(todo, one.action);
+        if (one["break"]) stop = true;
+      }
+    }
+    if (one["case"] == "timeout" && core.getFlag("type") == -1) {
+      found = true;
+      core.push(todo, one.action);
+      if (one["break"]) stop = true;
+    }
+  });
+  if (found) {
+    core.insertAction(todo);
+    return true;
+  }
+  return false;
+};
 
 events.prototype._precompile_wait = function (data) {
-    if (data.data) {
-        data.data.forEach(function (v) {
-            if (v.px != null) v.px = this.__precompile_array(v.px);
-            if (v.py != null) v.py = this.__precompile_array(v.py);
-            if (v.condition != null) v.condition = this.__precompile_array(v.condition);
-            v.action = this.precompile(v.action);
-        }, this);
-    }
-    return data;
-}
+  if (data.data) {
+    data.data.forEach(function (v) {
+      if (v.px != null) v.px = this.__precompile_array(v.px);
+      if (v.py != null) v.py = this.__precompile_array(v.py);
+      if (v.condition != null)
+        v.condition = this.__precompile_array(v.condition);
+      v.action = this.precompile(v.action);
+    }, this);
+  }
+  return data;
+};
 
 events.prototype._action_waitAsync = function (data, x, y, prefix) {
-    var test = window.setInterval(function () {
-        if (!core.hasAsync()
-            && (data.excludeAnimates || core.isReplaying() || core.getPlayingAnimates().length == 0)
-            && (!data.includeSounds || core.isReplaying() || core.getPlayingSounds().length == 0)) {
-            clearInterval(test);
-            core.doAction();
-        }
-    }, 50 / core.status.replay.speed);
-}
+  var test = window.setInterval(function () {
+    if (
+      !core.hasAsync() &&
+      (data.excludeAnimates ||
+        core.isReplaying() ||
+        core.getPlayingAnimates().length == 0) &&
+      (!data.includeSounds ||
+        core.isReplaying() ||
+        core.getPlayingSounds().length == 0)
+    ) {
+      clearInterval(test);
+      core.doAction();
+    }
+  }, 50 / core.status.replay.speed);
+};
 
 events.prototype._action_stopAsync = function (data, x, y, prefix) {
-    core.stopAsync();
-    core.doAction();
-}
+  core.stopAsync();
+  core.doAction();
+};
 
 events.prototype._action_callBook = function (data, x, y, prefix) {
-    if (core.isReplaying() || !core.hasItem('book')) {
-        core.doAction();
-    }
-    else {
-        var e = core.clone(core.status.event.data);
-        core.ui.closePanel();
-        core.openBook();
-        core.status.event.interval = e;
-    }
-}
-
-events.prototype._action_callSave = function (data, x, y, prefix) {
-    if (core.isReplaying() || core.hasFlag("__events__")) {
-        core.removeFlag("__events__");
-        core.doAction();
-    }
-    else {
-        var e = core.clone(core.status.event.data);
-        core.ui.closePanel();
-        var forbidSave = core.hasFlag('__forbidSave__');
-        core.removeFlag('__forbidSave__');
-        core.save();
-        if (forbidSave) core.setFlag('__forbidSave__', true);
-        core.status.event.interval = e;
-    }
-}
-
-events.prototype._action_autoSave = function (data, x, y, prefix) {
-    var forbidSave = core.hasFlag('__forbidSave__');
-    core.removeFlag('__forbidSave__');
-    core.autosave(data.removeLast);
-    if (forbidSave) core.setFlag('__forbidSave__', true);
+  if (core.isReplaying() || !core.hasItem("book")) {
     core.doAction();
-}
-
-events.prototype._action_forbidSave = function (data, x, y, prefix) {
-    core.setFlag('__forbidSave__', data.forbid || null);
-    core.doAction();
-}
-
-events.prototype._action_callLoad = function (data, x, y, prefix) {
-    if (this.__action_checkReplaying()) return;
+  } else {
     var e = core.clone(core.status.event.data);
     core.ui.closePanel();
-    core.load();
+    core.openBook();
     core.status.event.interval = e;
-}
+  }
+};
+
+events.prototype._action_callSave = function (data, x, y, prefix) {
+  if (core.isReplaying() || core.hasFlag("__events__")) {
+    core.removeFlag("__events__");
+    core.doAction();
+  } else {
+    var e = core.clone(core.status.event.data);
+    core.ui.closePanel();
+    var forbidSave = core.hasFlag("__forbidSave__");
+    core.removeFlag("__forbidSave__");
+    core.save();
+    if (forbidSave) core.setFlag("__forbidSave__", true);
+    core.status.event.interval = e;
+  }
+};
+
+events.prototype._action_autoSave = function (data, x, y, prefix) {
+  var forbidSave = core.hasFlag("__forbidSave__");
+  core.removeFlag("__forbidSave__");
+  core.autosave(data.removeLast);
+  if (forbidSave) core.setFlag("__forbidSave__", true);
+  core.doAction();
+};
+
+events.prototype._action_forbidSave = function (data, x, y, prefix) {
+  core.setFlag("__forbidSave__", data.forbid || null);
+  core.doAction();
+};
+
+events.prototype._action_callLoad = function (data, x, y, prefix) {
+  if (this.__action_checkReplaying()) return;
+  var e = core.clone(core.status.event.data);
+  core.ui.closePanel();
+  core.load();
+  core.status.event.interval = e;
+};
 
 events.prototype._action_exit = function (data, x, y, prefix) {
-    this.setEvents([]);
-    core.doAction();
-}
+  this.setEvents([]);
+  core.doAction();
+};
 
 events.prototype._action_previewUI = function (data, x, y, prefix) {
-    this.insertAction(data.action);
-    core.doAction();
-}
+  this.insertAction(data.action);
+  core.doAction();
+};
 
 events.prototype._precompile_previewUI = function (data) {
-    data.action = this.precompile(data.action);
-    return data;
-}
+  data.action = this.precompile(data.action);
+  return data;
+};
 
 events.prototype.__action_doUIEvent = function (data) {
-    this.__action_doUIEvent_doOne(data);
-    var current = core.status.event.data.list[0];
-    while (current.todo.length > 0) {
-        data = current.todo[0];
-        if (this.__action_doUIEvent_doOne(current.todo[0]))
-            current.todo.shift();
-        else break;
-    }
-    core.doAction();
-}
+  this.__action_doUIEvent_doOne(data);
+  var current = core.status.event.data.list[0];
+  while (current.todo.length > 0) {
+    data = current.todo[0];
+    if (this.__action_doUIEvent_doOne(current.todo[0])) current.todo.shift();
+    else break;
+  }
+  core.doAction();
+};
 
 events.prototype.__action_doUIEvent_doOne = function (data) {
-    if (core.ui['_uievent_' + data.type]) {
-        core.ui['_uievent_' + data.type](data);
-        return true;
-    }
-    return false;
-}
+  if (core.ui["_uievent_" + data.type]) {
+    core.ui["_uievent_" + data.type](data);
+    return true;
+  }
+  return false;
+};
 
 events.prototype._action_clearMap = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_fillText = function (data, x, y, prefix) {
-    data.text = core.replaceText(data.text, prefix);
-    this.__action_doUIEvent(data);
-}
+  data.text = core.replaceText(data.text, prefix);
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_fillBoldText = function (data, x, y, prefix) {
-    data.text = core.replaceText(data.text, prefix);
-    this.__action_doUIEvent(data);
-}
+  data.text = core.replaceText(data.text, prefix);
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_fillRect = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_fillPolygon = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._precompile_fillPolygon = function (data) {
-    data.nodes = this.__precompile_array(data.nodes);
-    return data;
-}
+  data.nodes = this.__precompile_array(data.nodes);
+  return data;
+};
 
 events.prototype._action_strokeRect = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_strokePolygon = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._precompile_strokePolygon = function (data) {
-    data.nodes = this.__precompile_array(data.nodes);
-    return data;
-}
+  data.nodes = this.__precompile_array(data.nodes);
+  return data;
+};
 
 events.prototype._action_fillEllipse = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_strokeEllipse = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_fillArc = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_strokeArc = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_drawLine = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_drawArrow = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_setAttribute = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_setFilter = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_drawImage = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_drawIcon = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_drawSelector = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_drawBackground = function (data, x, y, prefix) {
-    this.__action_doUIEvent(data);
-}
+  this.__action_doUIEvent(data);
+};
 
 events.prototype._action_drawTextContent = function (data, x, y, prefix) {
-    data.text = core.replaceText(data.text, prefix);
-    this.__action_doUIEvent(data);
-}
+  data.text = core.replaceText(data.text, prefix);
+  this.__action_doUIEvent(data);
+};
 
 // ------ 点击状态栏图标所进行的一些操作 ------ //
 
 ////// 判断当前能否进入某个事件 //////
 events.prototype._checkStatus = function (name, fromUserAction, checkItem) {
-    if (fromUserAction && core.status.event.id == name) {
-        core.ui.closePanel();
-        return false;
-    }
-    if (fromUserAction && core.status.lockControl) return false;
-    if (checkItem && !core.hasItem(name)) {
-        core.playSound('操作失败');
-        core.drawTip("你没有" + core.material.items[name].name, name);
-        return false;
-    }
-    if (core.isMoving()) {
-        core.playSound('操作失败');
-        core.drawTip("请先停止勇士行动");
-        return false;
-    }
-    core.lockControl();
-    core.status.event.id = name;
-    return true;
-}
+  if (fromUserAction && core.status.event.id == name) {
+    core.ui.closePanel();
+    return false;
+  }
+  if (fromUserAction && core.status.lockControl) return false;
+  if (checkItem && !core.hasItem(name)) {
+    core.playSound("操作失败");
+    core.drawTip("你没有" + core.material.items[name].name, name);
+    return false;
+  }
+  if (core.isMoving()) {
+    core.playSound("操作失败");
+    core.drawTip("请先停止勇士行动");
+    return false;
+  }
+  core.lockControl();
+  core.status.event.id = name;
+  return true;
+};
 
 ////// 点击怪物手册时的打开操作 //////
 events.prototype.openBook = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    // 如果能恢复事件(从callBook事件触发)
-    if (core.status.event.id == 'book' && core.events.recoverEvents(core.status.event.interval))
-        return;
-    // 当前是book,且从“浏览地图”打开
-    if (core.status.event.id == 'book' && core.status.event.ui) {
-        core.status.boxAnimateObjs = [];
-        core.ui._drawViewMaps(core.status.event.ui);
-        return;
-    }
-    // 从“浏览地图”页面打开
-    if (core.status.event.id == 'viewMaps') {
-        fromUserAction = false;
-        core.status.event.ui = core.status.event.data;
-    }
-    if (!this._checkStatus('book', fromUserAction, true)) return;
-    core.playSound('打开界面');
-    core.useItem('book', true);
-}
+  if (core.isReplaying()) return;
+  // 如果能恢复事件(从callBook事件触发)
+  if (
+    core.status.event.id == "book" &&
+    core.events.recoverEvents(core.status.event.interval)
+  )
+    return;
+  // 当前是book,且从“浏览地图”打开
+  if (core.status.event.id == "book" && core.status.event.ui) {
+    core.status.boxAnimateObjs = [];
+    core.ui._drawViewMaps(core.status.event.ui);
+    return;
+  }
+  // 从“浏览地图”页面打开
+  if (core.status.event.id == "viewMaps") {
+    fromUserAction = false;
+    core.status.event.ui = core.status.event.data;
+  }
+  if (!this._checkStatus("book", fromUserAction, true)) return;
+  core.playSound("打开界面");
+  core.useItem("book", true);
+};
 
 ////// 点击楼层传送器时的打开操作 //////
 events.prototype.useFly = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    // 从“浏览地图”页面:尝试直接传送到该层
-    if (core.status.event.id == 'viewMaps') {
-        if (!core.hasItem('fly')) {
-            core.playSound('操作失败');
-            core.drawTip('你没有' + core.material.items['fly'].name, 'fly');
-        } else if (!core.canUseItem('fly') || (core.flags.flyNearStair && !core.nearStair())) {
-            core.playSound('操作失败');
-            core.drawTip('无法传送到当前层', 'fly');
-        } else {
-            core.flyTo(core.status.event.data.floorId);
-        }
-        return;
+  if (core.isReplaying()) return;
+  // 从“浏览地图”页面:尝试直接传送到该层
+  if (core.status.event.id == "viewMaps") {
+    if (!core.hasItem("fly")) {
+      core.playSound("操作失败");
+      core.drawTip("你没有" + core.material.items["fly"].name, "fly");
+    } else if (
+      !core.canUseItem("fly") ||
+      (core.flags.flyNearStair && !core.nearStair())
+    ) {
+      core.playSound("操作失败");
+      core.drawTip("无法传送到当前层", "fly");
+    } else {
+      core.flyTo(core.status.event.data.floorId);
     }
-
-    if (!this._checkStatus('fly', fromUserAction, true)) return;
-    if (core.flags.flyNearStair && !core.nearStair()) {
-        core.playSound('操作失败');
-        core.drawTip("只有在楼梯边才能使用" + core.material.items['fly'].name, 'fly');
-        core.unlockControl();
-        core.status.event.data = null;
-        core.status.event.id = null;
-        return;
-    }
-    if (!core.canUseItem('fly')) {
-        core.playSound('操作失败');
-        core.drawTip(core.material.items['fly'].name + "好像失效了", 'fly');
-        core.unlockControl();
-        core.status.event.data = null;
-        core.status.event.id = null;
-        return;
-    }
-    core.playSound('打开界面');
-    core.useItem('fly', true);
     return;
-}
+  }
+
+  if (!this._checkStatus("fly", fromUserAction, true)) return;
+  if (core.flags.flyNearStair && !core.nearStair()) {
+    core.playSound("操作失败");
+    core.drawTip(
+      "只有在楼梯边才能使用" + core.material.items["fly"].name,
+      "fly"
+    );
+    core.unlockControl();
+    core.status.event.data = null;
+    core.status.event.id = null;
+    return;
+  }
+  if (!core.canUseItem("fly")) {
+    core.playSound("操作失败");
+    core.drawTip(core.material.items["fly"].name + "好像失效了", "fly");
+    core.unlockControl();
+    core.status.event.data = null;
+    core.status.event.id = null;
+    return;
+  }
+  core.playSound("打开界面");
+  core.useItem("fly", true);
+  return;
+};
 
 events.prototype.flyTo = function (toId, callback) {
-    return this.eventdata.flyTo(toId, callback);
-}
+  return this.eventdata.flyTo(toId, callback);
+};
 
 ////// 点击装备栏时的打开操作 //////
 events.prototype.openEquipbox = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    if (!this._checkStatus('equipbox', fromUserAction)) return;
-    core.playSound('打开界面');
-    core.ui._drawEquipbox();
-}
+  if (core.isReplaying()) return;
+  if (!this._checkStatus("equipbox", fromUserAction)) return;
+  core.playSound("打开界面");
+  core.ui._drawEquipbox();
+};
 
 ////// 点击工具栏时的打开操作 //////
 events.prototype.openToolbox = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    if (!this._checkStatus('toolbox', fromUserAction)) return;
-    core.playSound('打开界面');
-    core.ui._drawToolbox();
-}
+  if (core.isReplaying()) return;
+  if (!this._checkStatus("toolbox", fromUserAction)) return;
+  core.playSound("打开界面");
+  core.ui._drawToolbox();
+};
 
 ////// 点击快捷商店按钮时的打开操作 //////
 events.prototype.openQuickShop = function (fromUserAction) {
-    if (core.isReplaying()) return;
+  if (core.isReplaying()) return;
 
-    if (Object.keys(core.status.shops).length == 0) {
-        core.playSound('操作失败');
-        core.drawTip("本游戏没有快捷商店!");
-        return;
+  if (Object.keys(core.status.shops).length == 0) {
+    core.playSound("操作失败");
+    core.drawTip("本游戏没有快捷商店!");
+    return;
+  }
+
+  // --- 如果只有一个商店,则直接打开之
+  if (Object.keys(core.status.shops).length == 1) {
+    var shopId = Object.keys(core.status.shops)[0];
+    if (core.status.event.id != null) return;
+    if (!core.canOpenShop(shopId)) {
+      core.playSound("操作失败");
+      core.drawTip("当前无法打开快捷商店!");
+      return;
     }
-
-    // --- 如果只有一个商店,则直接打开之
-    if (Object.keys(core.status.shops).length == 1) {
-        var shopId = Object.keys(core.status.shops)[0];
-        if (core.status.event.id != null) return;
-        if (!core.canOpenShop(shopId)) {
-            core.playSound('操作失败');
-            core.drawTip("当前无法打开快捷商店!");
-            return;
-        }
-        var message = core.canUseQuickShop(shopId);
-        if (message != null) {
-            core.playSound('操作失败');
-            core.drawTip(message);
-            return;
-        }
-        core.openShop(shopId, false);
-        return;
+    var message = core.canUseQuickShop(shopId);
+    if (message != null) {
+      core.playSound("操作失败");
+      core.drawTip(message);
+      return;
     }
+    core.openShop(shopId, false);
+    return;
+  }
 
-    if (!this._checkStatus('selectShop', fromUserAction)) return;
-    core.ui._drawQuickShop();
-}
+  if (!this._checkStatus("selectShop", fromUserAction)) return;
+  core.ui._drawQuickShop();
+};
 
 events.prototype.openKeyBoard = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    if (!this._checkStatus('keyBoard', fromUserAction)) return;
-    core.ui._drawKeyBoard();
-}
+  if (core.isReplaying()) return;
+  if (!this._checkStatus("keyBoard", fromUserAction)) return;
+  core.ui._drawKeyBoard();
+};
 
 ////// 点击保存按钮时的打开操作 //////
 events.prototype.save = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    if (core.hasFlag('__forbidSave__')) {
-        core.playSound('操作失败');
-        core.drawTip('当前禁止存档');
-        return;
-    }
-    if (core.status.event.id == 'save' && core.events.recoverEvents(core.status.event.interval))
-        return;
-    if (!this._checkStatus('save', fromUserAction)) return;
-    var saveIndex = core.saves.saveIndex;
-    var page = parseInt((saveIndex - 1) / 5), offset = saveIndex - 5 * page;
-    core.playSound('打开界面');
-    core.ui._drawSLPanel(10 * page + offset);
-}
+  if (core.isReplaying()) return;
+  if (core.hasFlag("__forbidSave__")) {
+    core.playSound("操作失败");
+    core.drawTip("当前禁止存档");
+    return;
+  }
+  if (
+    core.status.event.id == "save" &&
+    core.events.recoverEvents(core.status.event.interval)
+  )
+    return;
+  if (!this._checkStatus("save", fromUserAction)) return;
+  var saveIndex = core.saves.saveIndex;
+  var page = parseInt((saveIndex - 1) / 5),
+    offset = saveIndex - 5 * page;
+  core.playSound("打开界面");
+  core.ui._drawSLPanel(10 * page + offset);
+};
 
 ////// 点击读取按钮时的打开操作 //////
 events.prototype.load = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    var saveIndex = core.saves.saveIndex;
-    var page = parseInt((saveIndex - 1) / 5), offset = saveIndex - 5 * page;
-    // 游戏开始前读档
-    if (!core.isPlaying()) {
-        core.dom.startPanel.style.display = 'none';
-        core.clearStatus();
-        core.clearMap('all');
-        core.status.event = { 'id': 'load', 'data': null };
-        core.status.lockControl = true;
-        core.playSound('打开界面');
-        core.ui._drawSLPanel(10 * page + offset);
-        return;
-    }
-    if (core.status.event.id == 'load' && core.events.recoverEvents(core.status.event.interval))
-        return;
-    if (!this._checkStatus('load', fromUserAction)) return;
-    core.playSound('打开界面');
+  if (core.isReplaying()) return;
+  var saveIndex = core.saves.saveIndex;
+  var page = parseInt((saveIndex - 1) / 5),
+    offset = saveIndex - 5 * page;
+  // 游戏开始前读档
+  if (!core.isPlaying()) {
+    core.dom.startPanel.style.display = "none";
+    core.clearStatus();
+    core.clearMap("all");
+    core.status.event = { id: "load", data: null };
+    core.status.lockControl = true;
+    core.playSound("打开界面");
     core.ui._drawSLPanel(10 * page + offset);
-}
+    return;
+  }
+  if (
+    core.status.event.id == "load" &&
+    core.events.recoverEvents(core.status.event.interval)
+  )
+    return;
+  if (!this._checkStatus("load", fromUserAction)) return;
+  core.playSound("打开界面");
+  core.ui._drawSLPanel(10 * page + offset);
+};
 
 ////// 点击设置按钮时的操作 //////
 events.prototype.openSettings = function (fromUserAction) {
-    if (core.isReplaying()) return;
-    if (!this._checkStatus('settings', fromUserAction))
-        return;
-    core.playSound('打开界面');
-    core.ui._drawSettings();
-}
+  if (core.isReplaying()) return;
+  if (!this._checkStatus("settings", fromUserAction)) return;
+  core.playSound("打开界面");
+  core.ui._drawSettings();
+};
 
 // ------ 一些事件的具体执行过程 ------ //
 
 events.prototype.hasAsync = function () {
-    return Object.keys(core.animateFrame.asyncId).length > 0;
-}
+  return Object.keys(core.animateFrame.asyncId).length > 0;
+};
 
 ////// 立刻停止所有异步事件 //////
 events.prototype.stopAsync = function () {
-    var callbacks = [];
-    for (var id in core.animateFrame.asyncId) {
-        clearInterval(id);
-        callbacks.push(core.animateFrame.asyncId[id]);
-    }
-    core.animateFrame.asyncId = {};
-    callbacks.forEach(function (cb) {
-        if (cb && cb instanceof Function) cb();
-    });
-}
+  var callbacks = [];
+  for (var id in core.animateFrame.asyncId) {
+    clearInterval(id);
+    callbacks.push(core.animateFrame.asyncId[id]);
+  }
+  core.animateFrame.asyncId = {};
+  callbacks.forEach(function (cb) {
+    if (cb && cb instanceof Function) cb();
+  });
+};
 
 events.prototype.hasAsyncAnimate = function () {
-    return (core.status.animateObjs || []).length > 0;
-}
+  return (core.status.animateObjs || []).length > 0;
+};
 
 ////// 跟随 //////
 events.prototype.follow = function (name) {
-    name = core.getMappedName(name);
-    if (core.material.images.images[name]) {
-        core.status.hero.followers.push({ "name": name });
-        core.gatherFollowers();
-        core.clearMap('hero');
-        core.drawHero();
-    }
-    core.clearRouteFolding();
-}
+  name = core.getMappedName(name);
+  if (core.material.images.images[name]) {
+    core.status.hero.followers.push({ name: name });
+    core.gatherFollowers();
+    core.clearMap("hero");
+    core.drawHero();
+  }
+  core.clearRouteFolding();
+};
 
 ////// 取消跟随 //////
 events.prototype.unfollow = function (name) {
-    if (!name) {
-        core.status.hero.followers = [];
+  if (!name) {
+    core.status.hero.followers = [];
+  } else {
+    name = core.getMappedName(name);
+    for (var i = 0; i < core.status.hero.followers.length; i++) {
+      if (core.status.hero.followers[i].name == name) {
+        core.status.hero.followers.splice(i, 1);
+        break;
+      }
     }
-    else {
-        name = core.getMappedName(name);
-        for (var i = 0; i < core.status.hero.followers.length; i++) {
-            if (core.status.hero.followers[i].name == name) {
-                core.status.hero.followers.splice(i, 1);
-                break;
-            }
-        }
-    }
-    core.gatherFollowers();
-    core.clearMap('hero');
-    core.drawHero();
-    core.clearRouteFolding();
-}
+  }
+  core.gatherFollowers();
+  core.clearMap("hero");
+  core.drawHero();
+  core.clearRouteFolding();
+};
 
-events.prototype._updateValueByOperator = function (value, originValue, operator) {
-    switch (operator) {
-        case '+=': value = originValue + value; break;
-        case '-=': value = originValue - value; break;
-        case '*=': value = originValue * value; break;
-        case '/=': value = originValue / value; break;
-        case '**=': value = Math.pow(originValue, value); break;
-        case '//=': value = Math.trunc(originValue / value); break;
-        case '%=': value = originValue % value; break;
-        case 'min=': value = Math.min(originValue, value); break;
-        case 'max=': value = Math.max(originValue, value); break;
-        default: break;
-    }
-    return value;
-}
+events.prototype._updateValueByOperator = function (
+  value,
+  originValue,
+  operator
+) {
+  switch (operator) {
+    case "+=":
+      value = originValue + value;
+      break;
+    case "-=":
+      value = originValue - value;
+      break;
+    case "*=":
+      value = originValue * value;
+      break;
+    case "/=":
+      value = originValue / value;
+      break;
+    case "**=":
+      value = Math.pow(originValue, value);
+      break;
+    case "//=":
+      value = Math.trunc(originValue / value);
+      break;
+    case "%=":
+      value = originValue % value;
+      break;
+    case "min=":
+      value = Math.min(originValue, value);
+      break;
+    case "max=":
+      value = Math.max(originValue, value);
+      break;
+    default:
+      break;
+  }
+  return value;
+};
 
 ////// 数值操作 //////
 events.prototype.setValue = function (name, operator, value, prefix) {
-    value = this._updateValueByOperator(core.calValue(value, prefix), core.calValue(name, prefix), operator);
-    this._setValue_setStatus(name, value);
-    this._setValue_setBuff(name, value);
-    this._setValue_setItem(name, value);
-    this._setValue_setFlag(name, value);
-    this._setValue_setSwitch(name, value, prefix);
-    this._setValue_setTemp(name, value, prefix);
-    this._setValue_setGlobal(name, value);
-}
+  value = this._updateValueByOperator(
+    core.calValue(value, prefix),
+    core.calValue(name, prefix),
+    operator
+  );
+  this._setValue_setStatus(name, value);
+  this._setValue_setBuff(name, value);
+  this._setValue_setItem(name, value);
+  this._setValue_setFlag(name, value);
+  this._setValue_setSwitch(name, value, prefix);
+  this._setValue_setTemp(name, value, prefix);
+  this._setValue_setGlobal(name, value);
+};
 
 events.prototype._setValue_setStatus = function (name, value) {
-    if (name.indexOf("status:") !== 0) return;
-    core.setStatus(name.substring(7), value);
-}
+  if (name.indexOf("status:") !== 0) return;
+  core.setStatus(name.substring(7), value);
+};
 
 events.prototype._setValue_setBuff = function (name, value) {
-    if (name.indexOf('buff:') !== 0) return;
-    core.setBuff(name.substring(5), value);
-}
+  if (name.indexOf("buff:") !== 0) return;
+  core.setBuff(name.substring(5), value);
+};
 
 events.prototype._setValue_setItem = function (name, value) {
-    if (name.indexOf("item:") !== 0) return;
-    var itemId = name.substring(5), count = core.itemCount(itemId);
-    if (value > count) core.getItem(itemId, value - count);
-    else core.setItem(itemId, value);
-}
+  if (name.indexOf("item:") !== 0) return;
+  var itemId = name.substring(5),
+    count = core.itemCount(itemId);
+  if (value > count) core.getItem(itemId, value - count);
+  else core.setItem(itemId, value);
+};
 
 events.prototype._setValue_setFlag = function (name, value) {
-    if (name.indexOf("flag:") !== 0) return;
-    core.setFlag(name.substring(5), value);
-}
+  if (name.indexOf("flag:") !== 0) return;
+  core.setFlag(name.substring(5), value);
+};
 
 events.prototype._setValue_setSwitch = function (name, value, prefix) {
-    if (name.indexOf("switch:") !== 0) return;
-    core.setFlag((prefix || ":f@x@y") + "@" + name.substring(7), value);
-}
+  if (name.indexOf("switch:") !== 0) return;
+  core.setFlag((prefix || ":f@x@y") + "@" + name.substring(7), value);
+};
 
 events.prototype._setValue_setTemp = function (name, value) {
-    if (name.indexOf("temp:") !== 0) return;
-    core.setFlag("@temp@" + name.substring(5), value);
-}
+  if (name.indexOf("temp:") !== 0) return;
+  core.setFlag("@temp@" + name.substring(5), value);
+};
 
 events.prototype._setValue_setGlobal = function (name, value) {
-    if (name.indexOf("global:") !== 0) return;
-    core.setGlobal(name.substring(7), value);
-}
+  if (name.indexOf("global:") !== 0) return;
+  core.setGlobal(name.substring(7), value);
+};
 
 ////// 设置一个怪物属性 //////
-events.prototype.setEnemy = function (id, name, value, operator, prefix, norefresh) {
-    if (!core.hasFlag('enemyInfo')) {
-        core.setFlag('enemyInfo', {});
-    }
-    var enemyInfo = core.getFlag('enemyInfo');
-    if (!enemyInfo[id]) enemyInfo[id] = {};
-    if (typeof value === 'string' && name == 'name') value = value.replace(/\r/g, '\\r');
-    value = this._updateValueByOperator(core.calValue(value, prefix), (core.material.enemys[id] || {})[name], operator);
-    enemyInfo[id][name] = value;
-    (core.material.enemys[id] || {})[name] = core.clone(value);
-    if (!norefresh) core.updateStatusBar();
-}
+events.prototype.setEnemy = function (
+  id,
+  name,
+  value,
+  operator,
+  prefix,
+  norefresh
+) {
+  if (!core.hasFlag("enemyInfo")) {
+    core.setFlag("enemyInfo", {});
+  }
+  var enemyInfo = core.getFlag("enemyInfo");
+  if (!enemyInfo[id]) enemyInfo[id] = {};
+  if (typeof value === "string" && name == "name")
+    value = value.replace(/\r/g, "\\r");
+  value = this._updateValueByOperator(
+    core.calValue(value, prefix),
+    (core.material.enemys[id] || {})[name],
+    operator
+  );
+  enemyInfo[id][name] = value;
+  (core.material.enemys[id] || {})[name] = core.clone(value);
+  if (!norefresh) core.updateStatusBar();
+};
 
 ////// 设置某个点上的怪物属性 //////
-events.prototype.setEnemyOnPoint = function (x, y, floorId, name, value, operator, prefix, norefresh) {
-    floorId = floorId || core.status.floorId;
-    var block = core.getBlock(x, y, floorId);
-    if (block == null) return;
-    if (block.event.cls.indexOf('enemy') != 0) return;
-    var enemy = core.material.enemys[block.event.id];
-    if (enemy == null) return;
-    if (typeof value === 'string' && name == 'name') value = value.replaceAll(/\r/g, '\\r');
-    value = this._updateValueByOperator(core.calValue(value, prefix), core.getEnemyValue(enemy, name, x, y, floorId), operator);
-    flags.enemyOnPoint = flags.enemyOnPoint || {};
-    flags.enemyOnPoint[floorId] = flags.enemyOnPoint[floorId] || {};
-    flags.enemyOnPoint[floorId][x + "," + y] = flags.enemyOnPoint[floorId][x + "," + y] || {};
-    flags.enemyOnPoint[floorId][x + "," + y][name] = value;
-    if (!norefresh) core.updateStatusBar();
-}
+events.prototype.setEnemyOnPoint = function (
+  x,
+  y,
+  floorId,
+  name,
+  value,
+  operator,
+  prefix,
+  norefresh
+) {
+  floorId = floorId || core.status.floorId;
+  var block = core.getBlock(x, y, floorId);
+  if (block == null) return;
+  if (block.event.cls.indexOf("enemy") != 0) return;
+  var enemy = core.material.enemys[block.event.id];
+  if (enemy == null) return;
+  if (typeof value === "string" && name == "name")
+    value = value.replaceAll(/\r/g, "\\r");
+  value = this._updateValueByOperator(
+    core.calValue(value, prefix),
+    core.getEnemyValue(enemy, name, x, y, floorId),
+    operator
+  );
+  flags.enemyOnPoint = flags.enemyOnPoint || {};
+  flags.enemyOnPoint[floorId] = flags.enemyOnPoint[floorId] || {};
+  flags.enemyOnPoint[floorId][x + "," + y] =
+    flags.enemyOnPoint[floorId][x + "," + y] || {};
+  flags.enemyOnPoint[floorId][x + "," + y][name] = value;
+  if (!norefresh) core.updateStatusBar();
+};
 
 ////// 重置某个点上的怪物属性 //////
 events.prototype.resetEnemyOnPoint = function (x, y, floorId, norefresh) {
-    delete ((flags.enemyOnPoint || {})[floorId || core.status.floorId] || {})[x + "," + y];
-    if (!norefresh) core.updateStatusBar();
-}
+  delete ((flags.enemyOnPoint || {})[floorId || core.status.floorId] || {})[
+    x + "," + y
+  ];
+  if (!norefresh) core.updateStatusBar();
+};
 
 ////// 将某个点上已经设置的怪物属性移动到其他点 //////
-events.prototype.moveEnemyOnPoint = function (fromX, fromY, toX, toY, floorId, norefresh) {
-    floorId = floorId || core.status.floorId;
-    if (((flags.enemyOnPoint || {})[floorId] || {})[fromX + "," + fromY]) {
-        flags.enemyOnPoint[floorId][toX + "," + toY] = flags.enemyOnPoint[floorId][fromX + "," + fromY];
-        delete flags.enemyOnPoint[floorId][fromX + "," + fromY];
-        if (!norefresh) core.updateStatusBar();
-    }
-}
+events.prototype.moveEnemyOnPoint = function (
+  fromX,
+  fromY,
+  toX,
+  toY,
+  floorId,
+  norefresh
+) {
+  floorId = floorId || core.status.floorId;
+  if (((flags.enemyOnPoint || {})[floorId] || {})[fromX + "," + fromY]) {
+    flags.enemyOnPoint[floorId][toX + "," + toY] =
+      flags.enemyOnPoint[floorId][fromX + "," + fromY];
+    delete flags.enemyOnPoint[floorId][fromX + "," + fromY];
+    if (!norefresh) core.updateStatusBar();
+  }
+};
 
 ////// 设置楼层属性 //////
 events.prototype.setFloorInfo = function (name, value, floorId, prefix) {
-    floorId = floorId || core.status.floorId;
-    core.status.maps[floorId][name] = value;
-    core.updateStatusBar();
-}
+  floorId = floorId || core.status.floorId;
+  core.status.maps[floorId][name] = value;
+  core.updateStatusBar();
+};
 
 ////// 设置全塔属性 //////
 events.prototype.setGlobalAttribute = function (name, value) {
-    if (typeof value == 'string') {
-        if ((value.charAt(0) == '"' && value.charAt(value.length - 1) == '"')
-            || (value.charAt(0) == "'" && value.charAt(value.length - 1) == "'"))
-            value = value.substring(1, value.length - 1);
-        // --- 检查 []
-        if (value.charAt(0) == '[' && value.charAt(value.length - 1) == ']')
-            value = eval(value);
-        // --- 检查颜色
-        if (/^[0-9 ]+,[0-9 ]+,[0-9 ]+(,[0-9. ]+)?$/.test(value)) {
-            value = 'rgba(' + value + ')';
-        }
+  if (typeof value == "string") {
+    if (
+      (value.charAt(0) == '"' && value.charAt(value.length - 1) == '"') ||
+      (value.charAt(0) == "'" && value.charAt(value.length - 1) == "'")
+    )
+      value = value.substring(1, value.length - 1);
+    // --- 检查 []
+    if (value.charAt(0) == "[" && value.charAt(value.length - 1) == "]")
+      value = eval(value);
+    // --- 检查颜色
+    if (/^[0-9 ]+,[0-9 ]+,[0-9 ]+(,[0-9. ]+)?$/.test(value)) {
+      value = "rgba(" + value + ")";
     }
-    core.status.globalAttribute[name] = value;
-    core.setFlag('globalAttribute', core.status.globalAttribute);
-    core.resize();
-}
+  }
+  core.status.globalAttribute[name] = value;
+  core.setFlag("globalAttribute", core.status.globalAttribute);
+  core.resize();
+};
 
 ////// 设置全局开关 //////
 events.prototype.setGlobalFlag = function (name, value) {
-    var flags = core.getFlag("globalFlags", {});
-    if (name.startsWith('s:')) {
-        name = name.substring(2);
-        var statusBarItems = core.flags.statusBarItems.filter(function (v) { return v != name; });
-        if (value) statusBarItems.push(name);
-        core.flags.statusBarItems = statusBarItems;
-        flags.statusBarItems = core.clone(statusBarItems);
-    } else {
-        flags[name] = core.flags[name] = value;
-    }
-    core.setFlag("globalFlags", flags);
-    core.resize();
-    if (name == 'blurFg')
-        core.redrawMap();
-}
+  var flags = core.getFlag("globalFlags", {});
+  if (name.startsWith("s:")) {
+    name = name.substring(2);
+    var statusBarItems = core.flags.statusBarItems.filter(function (v) {
+      return v != name;
+    });
+    if (value) statusBarItems.push(name);
+    core.flags.statusBarItems = statusBarItems;
+    flags.statusBarItems = core.clone(statusBarItems);
+  } else {
+    flags[name] = core.flags[name] = value;
+  }
+  core.setFlag("globalFlags", flags);
+  core.resize();
+  if (name == "blurFg") core.redrawMap();
+};
 
 ////// 设置文件别名 //////
 events.prototype.setNameMap = function (name, value) {
-    if (!core.hasFlag('__nameMap__')) core.setFlag('__nameMap__', {});
-    flags.__nameMap__[name] = value;
-}
+  if (!core.hasFlag("__nameMap__")) core.setFlag("__nameMap__", {});
+  flags.__nameMap__[name] = value;
+};
 
 ////// 设置剧情文本的属性 //////
 events.prototype.setTextAttribute = function (data) {
-    if (!core.isPlaying()) return;
-    ["position", "offset", "align", "bold", "titlefont", "textfont", "lineHeight", "time", "letterSpacing", "animateTime"].forEach(function (t) {
-        if (data[t] != null) core.status.textAttribute[t] = data[t];
-    });
-    ["background", "title", "text"].forEach(function (t) {
-        if ((data[t] instanceof Array) && data[t].length >= 3) {
-            if (data[t].length == 3) data[t].push(1);
-            core.status.textAttribute[t] = data[t];
-        }
-        if (t == 'background') {
-            var name = core.getMappedName(data[t]);
-            var img = core.material.images.images[name];
-            if (img && img.width == 192 && img.height == 128) {
-                core.status.textAttribute[t] = name;
-            }
-        }
-    });
-    if (main.mode == 'play') core.setFlag('textAttribute', core.status.textAttribute);
-}
-
-events.prototype.moveTextBox = function (code, loc, relative, moveMode, time, callback) {
-    var ctx = core.getContextByName('__text__' + code);
-    if (!ctx) {
-        if (callback) callback();
-        return;
+  if (!core.isPlaying()) return;
+  [
+    "position",
+    "offset",
+    "align",
+    "bold",
+    "titlefont",
+    "textfont",
+    "lineHeight",
+    "time",
+    "letterSpacing",
+    "animateTime",
+  ].forEach(function (t) {
+    if (data[t] != null) core.status.textAttribute[t] = data[t];
+  });
+  ["background", "title", "text"].forEach(function (t) {
+    if (data[t] instanceof Array && data[t].length >= 3) {
+      if (data[t].length == 3) data[t].push(1);
+      core.status.textAttribute[t] = data[t];
     }
-    var sx = parseInt(ctx.canvas.getAttribute('_text_left')) || 0;
-    var sy = parseInt(ctx.canvas.getAttribute('_text_top')) || 0;
-    var dx = relative ? loc[0] : (loc[0] - sx);
-    var dy = relative ? loc[1] : (loc[1] - sy);
-    var ox = parseInt(ctx.canvas.getAttribute('_left')) || 0;
-    var oy = parseInt(ctx.canvas.getAttribute('_top')) || 0;
-
-    if (!time) {
-        core.relocateCanvas(ctx, ox + dx, oy + dy);
-        ctx.canvas.setAttribute('_text_left', loc[0]);
-        ctx.canvas.setAttribute('_text_top', loc[1]);
-        if (callback) callback();
-        return;
+    if (t == "background") {
+      var name = core.getMappedName(data[t]);
+      var img = core.material.images.images[name];
+      if (img && img.width == 192 && img.height == 128) {
+        core.status.textAttribute[t] = name;
+      }
     }
+  });
+  if (main.mode == "play")
+    core.setFlag("textAttribute", core.status.textAttribute);
+};
 
-    var moveInfo = {
-        sx: sx, sy: sy, dx: dx, dy: dy, ox: ox, oy: oy,
-        moveMode: moveMode, time: time / Math.max(core.status.replay.speed, 1)
-    };
-    this._moveTextBox_moving(ctx, moveInfo, callback);
-}
+events.prototype.moveTextBox = function (
+  code,
+  loc,
+  relative,
+  moveMode,
+  time,
+  callback
+) {
+  var ctx = core.getContextByName("__text__" + code);
+  if (!ctx) {
+    if (callback) callback();
+    return;
+  }
+  var sx = parseInt(ctx.canvas.getAttribute("_text_left")) || 0;
+  var sy = parseInt(ctx.canvas.getAttribute("_text_top")) || 0;
+  var dx = relative ? loc[0] : loc[0] - sx;
+  var dy = relative ? loc[1] : loc[1] - sy;
+  var ox = parseInt(ctx.canvas.getAttribute("_left")) || 0;
+  var oy = parseInt(ctx.canvas.getAttribute("_top")) || 0;
+
+  if (!time) {
+    core.relocateCanvas(ctx, ox + dx, oy + dy);
+    ctx.canvas.setAttribute("_text_left", loc[0]);
+    ctx.canvas.setAttribute("_text_top", loc[1]);
+    if (callback) callback();
+    return;
+  }
+
+  var moveInfo = {
+    sx: sx,
+    sy: sy,
+    dx: dx,
+    dy: dy,
+    ox: ox,
+    oy: oy,
+    moveMode: moveMode,
+    time: time / Math.max(core.status.replay.speed, 1),
+  };
+  this._moveTextBox_moving(ctx, moveInfo, callback);
+};
 
 events.prototype._moveTextBox_moving = function (ctx, moveInfo, callback) {
-    var step = 0, steps = moveInfo.time / 10;
-    if (steps <= 0) steps = 1;
-    var moveFunc = core.applyEasing(moveInfo.moveMode);
-    var animate = setInterval(function () {
-        step++;
-        var dx = moveInfo.dx * moveFunc(step / steps);
-        var dy = moveInfo.dy * moveFunc(step / steps);
-        core.relocateCanvas(ctx, parseInt(moveInfo.ox + dx), parseInt(moveInfo.oy + dy));
-        ctx.canvas.setAttribute('_text_left', moveInfo.sx + dx);
-        ctx.canvas.setAttribute('_text_top', moveInfo.sy + dy);
-        if (step == steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            if (callback) callback();
-        }
-    }, 10);
+  var step = 0,
+    steps = moveInfo.time / 10;
+  if (steps <= 0) steps = 1;
+  var moveFunc = core.applyEasing(moveInfo.moveMode);
+  var animate = setInterval(function () {
+    step++;
+    var dx = moveInfo.dx * moveFunc(step / steps);
+    var dy = moveInfo.dy * moveFunc(step / steps);
+    core.relocateCanvas(
+      ctx,
+      parseInt(moveInfo.ox + dx),
+      parseInt(moveInfo.oy + dy)
+    );
+    ctx.canvas.setAttribute("_text_left", moveInfo.sx + dx);
+    ctx.canvas.setAttribute("_text_top", moveInfo.sy + dy);
+    if (step == steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      if (callback) callback();
+    }
+  }, 10);
 
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = callback;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = callback;
+};
 
 ////// 清除对话框 //////
 events.prototype.clearTextBox = function (code, callback) {
-    if (code == null) {
-        code = Object.keys(core.dymCanvas).filter(function (one) { return one.startsWith('__text__') })
-            .map(function (one) { return one.substring('__text__'.length); })
-    }
+  if (code == null) {
+    code = Object.keys(core.dymCanvas)
+      .filter(function (one) {
+        return one.startsWith("__text__");
+      })
+      .map(function (one) {
+        return one.substring("__text__".length);
+      });
+  }
 
-    if (!(code instanceof Array)) code = [code];
-    var index = 0;
-    var _work = function () {
-        if (index == code.length) {
-            if (callback) callback();
-            return;
-        }
-        var ctx = '__text__' + code[index++];
-        if (!core.getContextByName(ctx)) return _work();
-        core.ui._animateUI('hide', ctx, function () {
-            core.deleteCanvas(ctx);
-            _work();
-        });
-    };
-    _work();
-}
+  if (!(code instanceof Array)) code = [code];
+  var index = 0;
+  var _work = function () {
+    if (index == code.length) {
+      if (callback) callback();
+      return;
+    }
+    var ctx = "__text__" + code[index++];
+    if (!core.getContextByName(ctx)) return _work();
+    core.ui._animateUI("hide", ctx, function () {
+      core.deleteCanvas(ctx);
+      _work();
+    });
+  };
+  _work();
+};
 
 ////// 关门 //////
 events.prototype.closeDoor = function (x, y, id, callback) {
-    id = id || "";
-    if ((core.material.icons.animates[id] == null && core.material.icons.npc48[id] == null)
-        || core.getBlock(x, y) != null) {
-        if (callback) callback();
+  id = id || "";
+  if (
+    (core.material.icons.animates[id] == null &&
+      core.material.icons.npc48[id] == null) ||
+    core.getBlock(x, y) != null
+  ) {
+    if (callback) callback();
+    return;
+  }
+  var block = core.getBlockById(id);
+  var doorInfo = (block.event || {}).doorInfo;
+  if (!doorInfo) {
+    if (callback) callback();
+    return;
+  }
+
+  core.playSound(doorInfo.closeSound);
+  var blockInfo = core.getBlockInfo(block);
+  var speed = (doorInfo.time || 160) / 4;
+  blockInfo.posX = 3;
+  core.maps._drawBlockInfo(blockInfo, x, y);
+
+  var cb = function () {
+    core.setBlock(id, x, y);
+    core.showBlock(x, y);
+    if (callback) callback();
+  };
+
+  var animate = window.setInterval(
+    function () {
+      blockInfo.posX--;
+      if (blockInfo.posX < 0) {
+        clearInterval(animate);
+        delete core.animateFrame.asyncId[animate];
+        cb();
         return;
-    }
-    var block = core.getBlockById(id);
-    var doorInfo = (block.event || {}).doorInfo;
-    if (!doorInfo) {
-        if (callback) callback();
-        return;
-    }
+      }
+      core.maps._drawBlockInfo(blockInfo, x, y);
+    },
+    core.status.replay.speed == 24
+      ? 1
+      : speed / Math.max(core.status.replay.speed, 1)
+  );
 
-    core.playSound(doorInfo.closeSound);
-    var blockInfo = core.getBlockInfo(block);
-    var speed = (doorInfo.time || 160) / 4;
-    blockInfo.posX = 3;
-    core.maps._drawBlockInfo(blockInfo, x, y);
-
-    var cb = function () {
-        core.setBlock(id, x, y);
-        core.showBlock(x, y);
-        if (callback) callback();
-    }
-
-    var animate = window.setInterval(function () {
-        blockInfo.posX--;
-        if (blockInfo.posX < 0) {
-            clearInterval(animate);
-            delete core.animateFrame.asyncId[animate];
-            cb();
-            return;
-        }
-        core.maps._drawBlockInfo(blockInfo, x, y);
-    }, core.status.replay.speed == 24 ? 1 : speed / Math.max(core.status.replay.speed, 1));
-
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = cb;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = cb;
+};
 
 ////// 显示图片 //////
-events.prototype.showImage = function (code, image, sloc, loc, opacityVal, time, callback) {
-    var imageName = null;
-    if (typeof image == 'string') {
-        imageName = image;
-        if (image.endsWith(':x') || image.endsWith(':y') || image.endsWith(':o')) {
-            image = image.substring(0, image.length - 2);
-        }
-        image = core.getMappedName(image);
-        image = core.material.images.images[image];
+events.prototype.showImage = function (
+  code,
+  image,
+  sloc,
+  loc,
+  opacityVal,
+  time,
+  callback
+) {
+  var imageName = null;
+  if (typeof image == "string") {
+    imageName = image;
+    if (image.endsWith(":x") || image.endsWith(":y") || image.endsWith(":o")) {
+      image = image.substring(0, image.length - 2);
     }
-    if (!image) {
-        if (callback) callback();
-        return;
-    }
-    sloc = sloc || [];
-    var sx = core.calValue(sloc[0]) || 0, sy = core.calValue(sloc[1]) || 0;
-    var sw = core.calValue(sloc[2]), sh = core.calValue(sloc[3]);
-    if (sw == null) sw = image.width;
-    if (sh == null) sh = image.height;
-    loc = loc || [];
-    var x = core.calValue(loc[0]) || 0, y = core.calValue(loc[1]) || 0;
-    var w = core.calValue(loc[2]), h = core.calValue(loc[3]);
-    if (w == null) w = sw;
-    if (h == null) h = sh;
-    var zIndex = code + 100;
-    time = time || 0;
-    var name = "image" + zIndex;
-    var ctx = core.createCanvas(name, x, y, w, h, zIndex);
-    core.drawImage(ctx, imageName == null ? image : imageName, sx, sy, sw, sh, 0, 0, w, h);
-    if (time == 0) {
-        core.setOpacity(name, opacityVal);
-        if (callback) callback();
-        return;
-    }
-    core.setOpacity(name, 0);
-    this.moveImage(code, null, opacityVal, null, time, callback);
-}
+    image = core.getMappedName(image);
+    image = core.material.images.images[image];
+  }
+  if (!image) {
+    if (callback) callback();
+    return;
+  }
+  sloc = sloc || [];
+  var sx = core.calValue(sloc[0]) || 0,
+    sy = core.calValue(sloc[1]) || 0;
+  var sw = core.calValue(sloc[2]),
+    sh = core.calValue(sloc[3]);
+  if (sw == null) sw = image.width;
+  if (sh == null) sh = image.height;
+  loc = loc || [];
+  var x = core.calValue(loc[0]) || 0,
+    y = core.calValue(loc[1]) || 0;
+  var w = core.calValue(loc[2]),
+    h = core.calValue(loc[3]);
+  if (w == null) w = sw;
+  if (h == null) h = sh;
+  var zIndex = code + 100;
+  time = time || 0;
+  var name = "image" + zIndex;
+  var ctx = core.createCanvas(name, x, y, w, h, zIndex);
+  core.drawImage(
+    ctx,
+    imageName == null ? image : imageName,
+    sx,
+    sy,
+    sw,
+    sh,
+    0,
+    0,
+    w,
+    h
+  );
+  if (time == 0) {
+    core.setOpacity(name, opacityVal);
+    if (callback) callback();
+    return;
+  }
+  core.setOpacity(name, 0);
+  this.moveImage(code, null, opacityVal, null, time, callback);
+};
 
 ////// 隐藏图片 //////
 events.prototype.hideImage = function (code, time, callback) {
-    time = time || 0;
-    var name = "image" + (code + 100);
-    if (time == 0 || !core.dymCanvas[name]) {
-        core.deleteCanvas(name);
-        if (callback) callback();
-        return;
-    }
-    this.moveImage(code, null, 0, null, time, function () {
-        core.deleteCanvas(name);
-        if (callback) callback();
-    });
-}
+  time = time || 0;
+  var name = "image" + (code + 100);
+  if (time == 0 || !core.dymCanvas[name]) {
+    core.deleteCanvas(name);
+    if (callback) callback();
+    return;
+  }
+  this.moveImage(code, null, 0, null, time, function () {
+    core.deleteCanvas(name);
+    if (callback) callback();
+  });
+};
 
 ////// 移动图片 //////
-events.prototype.moveImage = function (code, to, opacityVal, moveMode, time, callback) {
-    to = to || [];
-    var name = "image" + (code + 100);
-    if (!core.dymCanvas[name]) {
-        if (callback) callback();
-        return;
-    }
-    var getOrDefault = function (a, b) {
-        a = core.calValue(a);
-        return a != null ? a : b;
-    }
-    var canvas = core.dymCanvas[name].canvas;
-    var fromX = parseFloat(canvas.getAttribute("_left")),
-        fromY = parseFloat(canvas.getAttribute("_top")),
-        toX = getOrDefault(to[0], fromX), toY = getOrDefault(to[1], fromY);
+events.prototype.moveImage = function (
+  code,
+  to,
+  opacityVal,
+  moveMode,
+  time,
+  callback
+) {
+  to = to || [];
+  var name = "image" + (code + 100);
+  if (!core.dymCanvas[name]) {
+    if (callback) callback();
+    return;
+  }
+  var getOrDefault = function (a, b) {
+    a = core.calValue(a);
+    return a != null ? a : b;
+  };
+  var canvas = core.dymCanvas[name].canvas;
+  var fromX = parseFloat(canvas.getAttribute("_left")),
+    fromY = parseFloat(canvas.getAttribute("_top")),
+    toX = getOrDefault(to[0], fromX),
+    toY = getOrDefault(to[1], fromY);
 
-    var opacity = parseFloat(canvas.style.opacity), toOpacity = getOrDefault(opacityVal, opacity);
+  var opacity = parseFloat(canvas.style.opacity),
+    toOpacity = getOrDefault(opacityVal, opacity);
 
-    if (!time) {
-        core.relocateCanvas(name, toX, toY);
-        core.setOpacity(toOpacity);
-        if (callback) callback();
-        return;
-    }
+  if (!time) {
+    core.relocateCanvas(name, toX, toY);
+    core.setOpacity(toOpacity);
+    if (callback) callback();
+    return;
+  }
 
-    this._moveImage_moving(name, {
-        fromX: fromX, fromY: fromY, toX: toX, toY: toY, opacity: opacity, toOpacity: toOpacity,
-        moveMode: moveMode, time: time / Math.max(core.status.replay.speed, 1)
-    }, callback)
-}
+  this._moveImage_moving(
+    name,
+    {
+      fromX: fromX,
+      fromY: fromY,
+      toX: toX,
+      toY: toY,
+      opacity: opacity,
+      toOpacity: toOpacity,
+      moveMode: moveMode,
+      time: time / Math.max(core.status.replay.speed, 1),
+    },
+    callback
+  );
+};
 
 events.prototype._moveImage_moving = function (name, moveInfo, callback) {
-    var per_time = 10, step = 0, steps = parseInt(moveInfo.time / per_time);
-    if (steps <= 0) steps = 1;
-    var fromX = moveInfo.fromX, fromY = moveInfo.fromY, toX = moveInfo.toX, toY = moveInfo.toY,
-        opacity = moveInfo.opacity, toOpacity = moveInfo.toOpacity;
-    var currX = fromX, currY = fromY, currOpacity = opacity;
-    var moveFunc = core.applyEasing(moveInfo.moveMode);
-    var animate = setInterval(function () {
-        step++;
-        currOpacity = opacity + (toOpacity - opacity) * moveFunc(step / steps);
-        currX = parseInt(fromX + (toX - fromX) * moveFunc(step / steps));
-        currY = parseInt(fromY + (toY - fromY) * moveFunc(step / steps));
-        core.setOpacity(name, currOpacity);
-        core.relocateCanvas(name, currX, currY);
-        if (step == steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            if (callback) callback();
-        }
-    }, per_time);
+  var per_time = 10,
+    step = 0,
+    steps = parseInt(moveInfo.time / per_time);
+  if (steps <= 0) steps = 1;
+  var fromX = moveInfo.fromX,
+    fromY = moveInfo.fromY,
+    toX = moveInfo.toX,
+    toY = moveInfo.toY,
+    opacity = moveInfo.opacity,
+    toOpacity = moveInfo.toOpacity;
+  var currX = fromX,
+    currY = fromY,
+    currOpacity = opacity;
+  var moveFunc = core.applyEasing(moveInfo.moveMode);
+  var animate = setInterval(function () {
+    step++;
+    currOpacity = opacity + (toOpacity - opacity) * moveFunc(step / steps);
+    currX = parseInt(fromX + (toX - fromX) * moveFunc(step / steps));
+    currY = parseInt(fromY + (toY - fromY) * moveFunc(step / steps));
+    core.setOpacity(name, currOpacity);
+    core.relocateCanvas(name, currX, currY);
+    if (step == steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      if (callback) callback();
+    }
+  }, per_time);
 
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = callback;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = callback;
+};
 
 ////// 旋转图片 //////
-events.prototype.rotateImage = function (code, center, angle, moveMode, time, callback) {
-    center = center || [];
-    var name = "image" + (code + 100);
-    if (!core.dymCanvas[name]) {
-        if (callback) callback();
-        return;
-    }
-    var canvas = core.dymCanvas[name].canvas;
-    var centerX = core.calValue(center[0]), centerY = core.calValue(center[1]);
+events.prototype.rotateImage = function (
+  code,
+  center,
+  angle,
+  moveMode,
+  time,
+  callback
+) {
+  center = center || [];
+  var name = "image" + (code + 100);
+  if (!core.dymCanvas[name]) {
+    if (callback) callback();
+    return;
+  }
+  var canvas = core.dymCanvas[name].canvas;
+  var centerX = core.calValue(center[0]),
+    centerY = core.calValue(center[1]);
 
-    var fromAngle = parseFloat(canvas.getAttribute('_angle')) || 0;
+  var fromAngle = parseFloat(canvas.getAttribute("_angle")) || 0;
 
-    if (!time) {
-        core.rotateCanvas(name, fromAngle + angle, centerX, centerY);
-        if (callback) callback();
-        return;
-    }
+  if (!time) {
+    core.rotateCanvas(name, fromAngle + angle, centerX, centerY);
+    if (callback) callback();
+    return;
+  }
 
-    var rotateInfo = {
-        fromAngle: fromAngle, angle: angle, centerX: centerX, centerY: centerY,
-        moveMode: moveMode, time: time / Math.max(core.status.replay.speed, 1)
-    }
-    this._rotateImage_rotating(name, rotateInfo, callback);
-}
+  var rotateInfo = {
+    fromAngle: fromAngle,
+    angle: angle,
+    centerX: centerX,
+    centerY: centerY,
+    moveMode: moveMode,
+    time: time / Math.max(core.status.replay.speed, 1),
+  };
+  this._rotateImage_rotating(name, rotateInfo, callback);
+};
 
 events.prototype._rotateImage_rotating = function (name, rotateInfo, callback) {
-    var per_time = 10, step = 0, steps = parseInt(rotateInfo.time / per_time);
-    if (steps <= 0) steps = 1;
-    var moveFunc = core.applyEasing(rotateInfo.moveMode);
-    var animate = setInterval(function () {
-        step++;
-        var currAngle = rotateInfo.fromAngle + rotateInfo.angle * moveFunc(step / steps);
-        core.rotateCanvas(name, currAngle, rotateInfo.centerX, rotateInfo.centerY);
-        if (step == steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            if (callback) callback();
-        }
-    }, per_time);
+  var per_time = 10,
+    step = 0,
+    steps = parseInt(rotateInfo.time / per_time);
+  if (steps <= 0) steps = 1;
+  var moveFunc = core.applyEasing(rotateInfo.moveMode);
+  var animate = setInterval(function () {
+    step++;
+    var currAngle =
+      rotateInfo.fromAngle + rotateInfo.angle * moveFunc(step / steps);
+    core.rotateCanvas(name, currAngle, rotateInfo.centerX, rotateInfo.centerY);
+    if (step == steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      if (callback) callback();
+    }
+  }, per_time);
 
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = callback;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = callback;
+};
 
 ////// 放缩一张图片 //////
-events.prototype.scaleImage = function (code, center, scale, moveMode, time, callback) {
-    center = center || [];
-    var name = "image" + (code + 100);
-    if (!core.dymCanvas[name]) {
-        if (callback) callback();
-        return;
-    }
-    var ctx = core.dymCanvas[name];
-    var currScale = 1.0;
-    if (ctx.canvas.hasAttribute('_scale')) {
-        currScale = parseFloat(ctx.canvas.getAttribute('_scale'));
-    }
-    var ratio = ctx.canvas.hasAttribute('isHD') ? core.domStyle.ratio : 1;
-    var width = ctx.canvas.width / ratio, height = ctx.canvas.height / ratio;
-    var currLeft = parseFloat(ctx.canvas.getAttribute("_left"));
-    var currTop = parseFloat(ctx.canvas.getAttribute("_top"));
-    var centerX = core.calValue(center[0]), centerY = core.calValue(center[1]);
-    if (centerX == null || centerY == null) {
-        centerX = currLeft + width * currScale / 2;
-        centerY = currTop + height * currScale / 2;
-    }
-    var scaleInfo = {
-        x: (currLeft - centerX) / currScale, y: (currTop - centerY) / currScale, centerX: centerX, centerY: centerY,
-        width: width, height: height, currScale: currScale, scale: scale, moveMode: moveMode, time: time
-    }
-    this._scaleImage_scale(ctx, scaleInfo, callback);
-}
+events.prototype.scaleImage = function (
+  code,
+  center,
+  scale,
+  moveMode,
+  time,
+  callback
+) {
+  center = center || [];
+  var name = "image" + (code + 100);
+  if (!core.dymCanvas[name]) {
+    if (callback) callback();
+    return;
+  }
+  var ctx = core.dymCanvas[name];
+  var currScale = 1.0;
+  if (ctx.canvas.hasAttribute("_scale")) {
+    currScale = parseFloat(ctx.canvas.getAttribute("_scale"));
+  }
+  var ratio = ctx.canvas.hasAttribute("isHD") ? core.domStyle.ratio : 1;
+  var width = ctx.canvas.width / ratio,
+    height = ctx.canvas.height / ratio;
+  var currLeft = parseFloat(ctx.canvas.getAttribute("_left"));
+  var currTop = parseFloat(ctx.canvas.getAttribute("_top"));
+  var centerX = core.calValue(center[0]),
+    centerY = core.calValue(center[1]);
+  if (centerX == null || centerY == null) {
+    centerX = currLeft + (width * currScale) / 2;
+    centerY = currTop + (height * currScale) / 2;
+  }
+  var scaleInfo = {
+    x: (currLeft - centerX) / currScale,
+    y: (currTop - centerY) / currScale,
+    centerX: centerX,
+    centerY: centerY,
+    width: width,
+    height: height,
+    currScale: currScale,
+    scale: scale,
+    moveMode: moveMode,
+    time: time,
+  };
+  this._scaleImage_scale(ctx, scaleInfo, callback);
+};
 
 events.prototype._scaleInfo_scale = function (ctx, scaleInfo, scale) {
-    core.resizeCanvas(ctx, scaleInfo.width * scale, scaleInfo.height * scale, true);
-    core.relocateCanvas(ctx, scaleInfo.centerX + scaleInfo.x * scale, scaleInfo.centerY + scaleInfo.y * scale);
-    ctx.canvas.setAttribute('_scale', scale);
-}
+  core.resizeCanvas(
+    ctx,
+    scaleInfo.width * scale,
+    scaleInfo.height * scale,
+    true
+  );
+  core.relocateCanvas(
+    ctx,
+    scaleInfo.centerX + scaleInfo.x * scale,
+    scaleInfo.centerY + scaleInfo.y * scale
+  );
+  ctx.canvas.setAttribute("_scale", scale);
+};
 
 events.prototype._scaleImage_scale = function (ctx, scaleInfo, callback) {
-    if (!scaleInfo.time) {
-        this._scaleInfo_scale(ctx, scaleInfo, scaleInfo.scale);
-        if (callback) callback();
-        return;
+  if (!scaleInfo.time) {
+    this._scaleInfo_scale(ctx, scaleInfo, scaleInfo.scale);
+    if (callback) callback();
+    return;
+  }
+
+  var per_time = 10,
+    step = 0,
+    steps = parseInt(scaleInfo.time / per_time);
+  if (steps <= 0) steps = 1;
+  var moveFunc = core.applyEasing(scaleInfo.moveMode);
+
+  var animate = setInterval(function () {
+    step++;
+    var scale =
+      scaleInfo.currScale +
+      (scaleInfo.scale - scaleInfo.currScale) * moveFunc(step / steps);
+    core.events._scaleInfo_scale(ctx, scaleInfo, scale);
+    if (step == steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      if (callback) callback();
     }
+  }, per_time);
 
-    var per_time = 10, step = 0, steps = parseInt(scaleInfo.time / per_time);
-    if (steps <= 0) steps = 1;
-    var moveFunc = core.applyEasing(scaleInfo.moveMode);
-
-    var animate = setInterval(function () {
-        step++;
-        var scale = scaleInfo.currScale + (scaleInfo.scale - scaleInfo.currScale) * moveFunc(step / steps);
-        core.events._scaleInfo_scale(ctx, scaleInfo, scale);
-        if (step == steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            if (callback) callback();
-        }
-    }, per_time);
-
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = callback;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = callback;
+};
 
 ////// 绘制或取消一张gif图片 //////
 events.prototype.showGif = function (name, x, y) {
-    name = core.getMappedName(name);
-    var image = core.material.images.images[name];
-    if (image) {
-        var gif = new Image();
-        gif.src = image.src;
-        gif.style.position = 'absolute';
-        gif.style.left = x * core.domStyle.scale + "px";
-        gif.style.top = y * core.domStyle.scale + "px";
-        gif.style.width = image.width * core.domStyle.scale + "px";
-        gif.style.height = image.height * core.domStyle.scale + "px";
-        core.dom.gif2.appendChild(gif);
-    }
-    else {
-        core.dom.gif2.innerHTML = "";
-    }
-}
+  name = core.getMappedName(name);
+  var image = core.material.images.images[name];
+  if (image) {
+    var gif = new Image();
+    gif.src = image.src;
+    gif.style.position = "absolute";
+    gif.style.left = x * core.domStyle.scale + "px";
+    gif.style.top = y * core.domStyle.scale + "px";
+    gif.style.width = image.width * core.domStyle.scale + "px";
+    gif.style.height = image.height * core.domStyle.scale + "px";
+    core.dom.gif2.appendChild(gif);
+  } else {
+    core.dom.gif2.innerHTML = "";
+  }
+};
 
 ////// 淡入淡出音乐 //////
 events.prototype.setVolume = function (value, time, callback) {
-    var set = function (value) {
-        core.musicStatus.designVolume = value;
-        if (core.musicStatus.playingBgm)
-            core.material.bgms[core.musicStatus.playingBgm].volume = core.musicStatus.userVolume * core.musicStatus.designVolume;
+  var set = function (value) {
+    core.musicStatus.designVolume = value;
+    if (core.musicStatus.playingBgm)
+      core.material.bgms[core.musicStatus.playingBgm].volume =
+        core.musicStatus.userVolume * core.musicStatus.designVolume;
+  };
+  if (!time || time < 100) {
+    set(value);
+    if (callback) callback();
+    return;
+  }
+  var currVolume = core.musicStatus.designVolume;
+  time /= Math.max(core.status.replay.speed, 1);
+  var per_time = 10,
+    step = 0,
+    steps = parseInt(time / per_time);
+  if (steps <= 0) steps = 1;
+  var animate = setInterval(function () {
+    step++;
+    set(currVolume + ((value - currVolume) * step) / steps);
+    if (step >= steps) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      if (callback) callback();
     }
-    if (!time || time < 100) {
-        set(value);
-        if (callback) callback();
-        return;
-    }
-    var currVolume = core.musicStatus.designVolume;
-    time /= Math.max(core.status.replay.speed, 1);
-    var per_time = 10, step = 0, steps = parseInt(time / per_time);
-    if (steps <= 0) steps = 1;
-    var animate = setInterval(function () {
-        step++;
-        set(currVolume + (value - currVolume) * step / steps);
-        if (step >= steps) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            if (callback) callback();
-        }
-    }, per_time);
+  }, per_time);
 
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = callback;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = callback;
+};
 
 ////// 画面震动 //////
 events.prototype.vibrate = function (direction, time, speed, power, callback) {
-    if (core.isReplaying()) {
-        if (callback) callback();
-        return;
+  if (core.isReplaying()) {
+    if (callback) callback();
+    return;
+  }
+  if (!time) time = 1000;
+  speed = speed || 10;
+  power = power || 10;
+  var shakeInfo = {
+    duration: parseInt(time / 10),
+    speed: speed,
+    power: power,
+    direction: 1,
+    shake: 0,
+  };
+  if (direction == "random") {
+    direction = ["horizontal", "vertical", "diagonal1", "diagonal2"][
+      Math.floor(Math.random() * 4)
+    ];
+  }
+  var cb = function () {
+    core.addGameCanvasTranslate(0, 0);
+    if (callback) callback();
+  };
+  var animate = setInterval(function () {
+    core.events._vibrate_update(shakeInfo);
+    switch (direction) {
+      case "vertical":
+        core.addGameCanvasTranslate(0, shakeInfo.shake);
+        break;
+      case "diagonal1":
+        core.addGameCanvasTranslate(shakeInfo.shake, shakeInfo.shake);
+        break;
+      case "diagonal2":
+        core.addGameCanvasTranslate(-shakeInfo.shake, shakeInfo.shake);
+        break;
+      default:
+        core.addGameCanvasTranslate(shakeInfo.shake, 0);
     }
-    if (!time) time = 1000;
-    speed = speed || 10;
-    power = power || 10;
-    var shakeInfo = { duration: parseInt(time / 10), speed: speed, power: power, direction: 1, shake: 0 };
-    if (direction == 'random') {
-        direction = ['horizontal', 'vertical', 'diagonal1', 'diagonal2'][Math.floor(Math.random() * 4)];
+    if (shakeInfo.duration === 0 && shakeInfo.shake == 0) {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      cb();
     }
-    var cb = function () {
-        core.addGameCanvasTranslate(0, 0);
-        if (callback) callback();
-    }
-    var animate = setInterval(function () {
-        core.events._vibrate_update(shakeInfo);
-        switch (direction) {
-            case 'vertical': core.addGameCanvasTranslate(0, shakeInfo.shake); break;
-            case 'diagonal1': core.addGameCanvasTranslate(shakeInfo.shake, shakeInfo.shake); break;
-            case 'diagonal2': core.addGameCanvasTranslate(-shakeInfo.shake, shakeInfo.shake); break;
-            default: core.addGameCanvasTranslate(shakeInfo.shake, 0);
-        }
-        if (shakeInfo.duration === 0 && shakeInfo.shake == 0) {
-            delete core.animateFrame.asyncId[animate];
-            clearInterval(animate);
-            cb();
-        }
-    }, 10);
+  }, 10);
 
-    core.animateFrame.lastAsyncId = animate;
-    core.animateFrame.asyncId[animate] = cb;
-}
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = cb;
+};
 
 events.prototype._vibrate_update = function (shakeInfo) {
-    if (shakeInfo.duration >= 1 || shakeInfo.shake != 0) {
-        var delta = shakeInfo.speed * shakeInfo.direction / 6;
-        if (shakeInfo.duration <= 1 && shakeInfo.shake * (shakeInfo.shake + delta) < 0) {
-            shakeInfo.shake = 0;
-        } else {
-            shakeInfo.shake += delta;
-        }
-        if (shakeInfo.shake > shakeInfo.power) {
-            shakeInfo.direction = -1;
-        }
-        if (shakeInfo.shake < -shakeInfo.power) {
-            shakeInfo.direction = 1;
-        }
-        if (shakeInfo.duration >= 1) {
-            shakeInfo.duration -= 1
-        }
+  if (shakeInfo.duration >= 1 || shakeInfo.shake != 0) {
+    var delta = (shakeInfo.speed * shakeInfo.direction) / 6;
+    if (
+      shakeInfo.duration <= 1 &&
+      shakeInfo.shake * (shakeInfo.shake + delta) < 0
+    ) {
+      shakeInfo.shake = 0;
+    } else {
+      shakeInfo.shake += delta;
     }
-}
+    if (shakeInfo.shake > shakeInfo.power) {
+      shakeInfo.direction = -1;
+    }
+    if (shakeInfo.shake < -shakeInfo.power) {
+      shakeInfo.direction = 1;
+    }
+    if (shakeInfo.duration >= 1) {
+      shakeInfo.duration -= 1;
+    }
+  }
+};
 
 /////// 使用事件让勇士移动。这个函数将不会触发任何事件 //////
 events.prototype.eventMoveHero = function (steps, time, callback) {
-    time = time || core.values.moveSpeed;
-    var step = 0, moveSteps = (steps || []).map(function (t) {
-        return [t.split(':')[0], parseInt(t.split(':')[1] || "1")];
-    }).filter(function (t) {
-        return ['up', 'down', 'left', 'right', 'forward', 'backward', 'leftup', 'leftdown', 'rightup', 'rightdown', 'speed'].indexOf(t[0]) >= 0
-            && !(t[0] == 'speed' && t[1] < 16);
-    });
-    core.status.heroMoving = -1;
-    var _run = function () {
-        var cb = function () {
-            core.status.heroMoving = 0;
-            core.drawHero();
-            if (callback) callback();
-        }
-
-        var animate = window.setInterval(function () {
-            if (moveSteps.length == 0) {
-                delete core.animateFrame.asyncId[animate];
-                clearInterval(animate);
-                cb();
-            }
-            else {
-                if (step == 0 && moveSteps[0][0] == 'speed' && moveSteps[0][1] >= 16) {
-                    time = moveSteps[0][1];
-                    moveSteps.shift();
-                    clearInterval(animate);
-                    delete core.animateFrame.asyncId[animate];
-                    _run();
-                }
-                else if (core.events._eventMoveHero_moving(++step, moveSteps))
-                    step = 0;
-            }
-        }, core.status.replay.speed == 24 ? 1 : time / 8 / core.status.replay.speed);
-
-        core.animateFrame.lastAsyncId = animate;
-        core.animateFrame.asyncId[animate] = cb;
-    }
-    _run();
-}
-
-events.prototype._eventMoveHero_moving = function (step, moveSteps) {
-    var curr = moveSteps[0];
-    var direction = curr[0], x = core.getHeroLoc('x'), y = core.getHeroLoc('y');
-    // ------ 前进/后退
-    var o = direction == 'backward' ? -1 : 1;
-    if (direction == 'forward' || direction == 'backward') direction = core.getHeroLoc('direction');
-    var faceDirection = direction;
-    if (direction == 'leftup' || direction == 'leftdown') faceDirection = 'left';
-    if (direction == 'rightup' || direction == 'rightdown') faceDirection = 'right';
-    core.setHeroLoc('direction', direction);
-    if (curr[1] <= 0) {
-        core.setHeroLoc('direction', faceDirection);
-        moveSteps.shift();
-        return true;
-    }
-    if (step <= 4) {
-        core.drawHero('leftFoot', 4 * o * step);
-    }
-    else if (step <= 8) {
-        core.drawHero('rightFoot', 4 * o * step);
-    }
-    if (step == 8) {
-        core.setHeroLoc('x', x + o * core.utils.scan2[direction].x, true);
-        core.setHeroLoc('y', y + o * core.utils.scan2[direction].y, true);
-        core.updateFollowers();
-        curr[1]--;
-        if (curr[1] <= 0) moveSteps.shift();
-        core.setHeroLoc('direction', faceDirection);
-        return true;
-    }
-    return false;
-}
-
-////// 勇士跳跃事件 //////
-events.prototype.jumpHero = function (ex, ey, time, callback) {
-    var sx = core.getHeroLoc('x'), sy = core.getHeroLoc('y');
-    if (ex == null) ex = sx;
-    if (ey == null) ey = sy;
-    var sx = core.status.hero.loc.x, sy = core.status.hero.loc.y;
-    if (!core.isset(ex)) ex = sx;
-    if (!core.isset(ey)) ey = sy;
-    var jumpInfo = core.maps.__generateJumpInfo(sx, sy, ex, ey, time || 500);
-    jumpInfo.icon = core.material.icons.hero[core.getHeroLoc('direction')];
-    jumpInfo.width = core.material.icons.hero.width || 32;
-    jumpInfo.height = core.material.icons.hero.height;
-
-    this._jumpHero_doJump(jumpInfo, callback);
-}
-
-events.prototype._jumpHero_doJump = function (jumpInfo, callback) {
+  time = time || core.values.moveSpeed;
+  var step = 0,
+    moveSteps = (steps || [])
+      .map(function (t) {
+        return [t.split(":")[0], parseInt(t.split(":")[1] || "1")];
+      })
+      .filter(function (t) {
+        return (
+          [
+            "up",
+            "down",
+            "left",
+            "right",
+            "forward",
+            "backward",
+            "leftup",
+            "leftdown",
+            "rightup",
+            "rightdown",
+            "speed",
+          ].indexOf(t[0]) >= 0 && !(t[0] == "speed" && t[1] < 16)
+        );
+      });
+  core.status.heroMoving = -1;
+  var _run = function () {
     var cb = function () {
-        core.setHeroLoc('x', jumpInfo.ex);
-        core.setHeroLoc('y', jumpInfo.ey);
-        core.status.heroMoving = 0;
-        core.drawHero();
-        if (callback) callback();
-    }
+      core.status.heroMoving = 0;
+      core.drawHero();
+      if (callback) callback();
+    };
 
-    core.status.heroMoving = -1;
-    var animate = window.setInterval(function () {
-        if (jumpInfo.jump_count > 0)
-            core.events._jumpHero_jumping(jumpInfo)
-        else {
-            delete core.animateFrame.asyncId[animate];
+    var animate = window.setInterval(
+      function () {
+        if (moveSteps.length == 0) {
+          delete core.animateFrame.asyncId[animate];
+          clearInterval(animate);
+          cb();
+        } else {
+          if (
+            step == 0 &&
+            moveSteps[0][0] == "speed" &&
+            moveSteps[0][1] >= 16
+          ) {
+            time = moveSteps[0][1];
+            moveSteps.shift();
             clearInterval(animate);
-            cb();
+            delete core.animateFrame.asyncId[animate];
+            _run();
+          } else if (core.events._eventMoveHero_moving(++step, moveSteps))
+            step = 0;
         }
-    }, jumpInfo.per_time);
+      },
+      core.status.replay.speed == 24 ? 1 : time / 8 / core.status.replay.speed
+    );
 
     core.animateFrame.lastAsyncId = animate;
     core.animateFrame.asyncId[animate] = cb;
-}
+  };
+  _run();
+};
+
+events.prototype._eventMoveHero_moving = function (step, moveSteps) {
+  var curr = moveSteps[0];
+  var direction = curr[0],
+    x = core.getHeroLoc("x"),
+    y = core.getHeroLoc("y");
+  // ------ 前进/后退
+  var o = direction == "backward" ? -1 : 1;
+  if (direction == "forward" || direction == "backward")
+    direction = core.getHeroLoc("direction");
+  var faceDirection = direction;
+  if (direction == "leftup" || direction == "leftdown") faceDirection = "left";
+  if (direction == "rightup" || direction == "rightdown")
+    faceDirection = "right";
+  core.setHeroLoc("direction", direction);
+  if (curr[1] <= 0) {
+    core.setHeroLoc("direction", faceDirection);
+    moveSteps.shift();
+    return true;
+  }
+  if (step <= 4) {
+    core.drawHero("leftFoot", 4 * o * step);
+  } else if (step <= 8) {
+    core.drawHero("rightFoot", 4 * o * step);
+  }
+  if (step == 8) {
+    core.setHeroLoc("x", x + o * core.utils.scan2[direction].x, true);
+    core.setHeroLoc("y", y + o * core.utils.scan2[direction].y, true);
+    core.updateFollowers();
+    curr[1]--;
+    if (curr[1] <= 0) moveSteps.shift();
+    core.setHeroLoc("direction", faceDirection);
+    return true;
+  }
+  return false;
+};
+
+////// 勇士跳跃事件 //////
+events.prototype.jumpHero = function (ex, ey, time, callback) {
+  var sx = core.getHeroLoc("x"),
+    sy = core.getHeroLoc("y");
+  if (ex == null) ex = sx;
+  if (ey == null) ey = sy;
+  var sx = core.status.hero.loc.x,
+    sy = core.status.hero.loc.y;
+  if (!core.isset(ex)) ex = sx;
+  if (!core.isset(ey)) ey = sy;
+  var jumpInfo = core.maps.__generateJumpInfo(sx, sy, ex, ey, time || 500);
+  jumpInfo.icon = core.material.icons.hero[core.getHeroLoc("direction")];
+  jumpInfo.width = core.material.icons.hero.width || 32;
+  jumpInfo.height = core.material.icons.hero.height;
+
+  this._jumpHero_doJump(jumpInfo, callback);
+};
+
+events.prototype._jumpHero_doJump = function (jumpInfo, callback) {
+  var cb = function () {
+    core.setHeroLoc("x", jumpInfo.ex);
+    core.setHeroLoc("y", jumpInfo.ey);
+    core.status.heroMoving = 0;
+    core.drawHero();
+    if (callback) callback();
+  };
+
+  core.status.heroMoving = -1;
+  var animate = window.setInterval(function () {
+    if (jumpInfo.jump_count > 0) core.events._jumpHero_jumping(jumpInfo);
+    else {
+      delete core.animateFrame.asyncId[animate];
+      clearInterval(animate);
+      cb();
+    }
+  }, jumpInfo.per_time);
+
+  core.animateFrame.lastAsyncId = animate;
+  core.animateFrame.asyncId[animate] = cb;
+};
 
 events.prototype._jumpHero_jumping = function (jumpInfo) {
-    core.clearMap('hero');
-    core.maps.__updateJumpInfo(jumpInfo);
-    var x = core.getHeroLoc('x'),
-        y = core.getHeroLoc('y');
-    var nowx = jumpInfo.px, nowy = jumpInfo.py, width = jumpInfo.width || 32, height = jumpInfo.height;
-    core.drawHero('stop', { x: nowx - 32 * x, y: nowy - 32 * y });
-}
+  core.clearMap("hero");
+  core.maps.__updateJumpInfo(jumpInfo);
+  var x = core.getHeroLoc("x"),
+    y = core.getHeroLoc("y");
+  var nowx = jumpInfo.px,
+    nowy = jumpInfo.py,
+    width = jumpInfo.width || 32,
+    height = jumpInfo.height;
+  core.drawHero("stop", { x: nowx - 32 * x, y: nowy - 32 * y });
+};
 
 ////// 设置角色行走图 //////
 events.prototype.setHeroIcon = function (name, noDraw) {
-    name = core.getMappedName(name);
-    var img = core.material.images.images[name];
-    if (!img) {
-        console.error("找不到图片: " + img);
-        return;
-    }
-    if (core.material.images.hero == img) return;
-    core.status.hero.image = name;
-    core.material.images.hero = img;
-    core.material.icons.hero.width = img.width / 4;
-    core.material.icons.hero.height = img.height / 4;
-    core.control.updateHeroIcon(name);
-    if (!noDraw) core.drawHero();
-}
+  name = core.getMappedName(name);
+  var img = core.material.images.images[name];
+  if (!img) {
+    console.error("找不到图片: " + img);
+    return;
+  }
+  if (core.material.images.hero == img) return;
+  core.status.hero.image = name;
+  core.material.images.hero = img;
+  core.material.icons.hero.width = img.width / 4;
+  core.material.icons.hero.height = img.height / 4;
+  core.control.updateHeroIcon(name);
+  if (!noDraw) core.drawHero();
+};
 
 ////// 检查升级事件 //////
 events.prototype.checkLvUp = function () {
-    var actions = [];
-    while (true) {
-        var next = this._checkLvUp_check();
-        if (next == null) break;
-        actions = actions.concat(next);
-    }
-    if (actions.length > 0) core.insertAction(actions);
-}
+  var actions = [];
+  while (true) {
+    var next = this._checkLvUp_check();
+    if (next == null) break;
+    actions = actions.concat(next);
+  }
+  if (actions.length > 0) core.insertAction(actions);
+};
 
 events.prototype._checkLvUp_check = function () {
-    if (core.flags.statusBarItems.indexOf('enableLevelUp') < 0 || !core.firstData.levelUp
-        || core.status.hero.lv >= core.firstData.levelUp.length) return null;
-    // 计算下一个所需要的数值
-    var next = (core.firstData.levelUp[core.status.hero.lv] || {});
-    var need = core.calValue(next.need);
-    if (need == null) return null;
-    if (core.status.hero.exp >= need) {
-        // 升级
-        core.status.hero.lv++;
-        if (next.clear) core.status.hero.exp -= need;
-        return next.action || [];
-    }
+  if (
+    core.flags.statusBarItems.indexOf("enableLevelUp") < 0 ||
+    !core.firstData.levelUp ||
+    core.status.hero.lv >= core.firstData.levelUp.length
+  )
     return null;
-}
+  // 计算下一个所需要的数值
+  var next = core.firstData.levelUp[core.status.hero.lv] || {};
+  var need = core.calValue(next.need);
+  if (need == null) return null;
+  if (core.status.hero.exp >= need) {
+    // 升级
+    core.status.hero.lv++;
+    if (next.clear) core.status.hero.exp -= need;
+    return next.action || [];
+  }
+  return null;
+};
 
 ////// 尝试使用道具 //////
 events.prototype.tryUseItem = function (itemId) {
-    if (itemId == 'book') {
-        core.ui.closePanel();
-        return core.openBook(false);
-    }
-    if (itemId == 'fly') {
-        core.ui.closePanel();
-        return core.useFly(false);
-    }
-    if (itemId == 'centerFly') {
-        core.ui.closePanel();
-        return core.ui._drawCenterFly();
-    }
-    if (core.canUseItem(itemId)) {
-        core.ui.closePanel();
-        core.useItem(itemId);
-    } else {
-        core.playSound('操作失败');
-        core.drawTip("当前无法使用" + core.material.items[itemId].name, itemId);
-    }
-}
+  if (itemId == "book") {
+    core.ui.closePanel();
+    return core.openBook(false);
+  }
+  if (itemId == "fly") {
+    core.ui.closePanel();
+    return core.useFly(false);
+  }
+  if (itemId == "centerFly") {
+    core.ui.closePanel();
+    return core.ui._drawCenterFly();
+  }
+  if (core.canUseItem(itemId)) {
+    core.ui.closePanel();
+    core.useItem(itemId);
+  } else {
+    core.playSound("操作失败");
+    core.drawTip("当前无法使用" + core.material.items[itemId].name, itemId);
+  }
+};
diff --git a/libs/loader.js b/libs/loader.js
index e00620d..b81aba2 100644
--- a/libs/loader.js
+++ b/libs/loader.js
@@ -1,606 +1,818 @@
-
 /*
 loader.js:负责对资源的加载
 
  */
 "use strict";
 
-function loader () {
-    this._init();
+function loader() {
+  this._init();
 }
 
-loader.prototype._init = function () {
-
-}
+loader.prototype._init = function () {};
 
 ////// 设置加载进度条进度 //////
 loader.prototype._setStartProgressVal = function (val) {
-    core.dom.startTopProgress.style.width = val + '%';
-}
+  core.dom.startTopProgress.style.width = val + "%";
+};
 
 ////// 设置加载进度条提示文字 //////
 loader.prototype._setStartLoadTipText = function (text) {
-    core.dom.startTopLoadTips.innerText = text;
-}
+  core.dom.startTopLoadTips.innerText = text;
+};
 
 loader.prototype._load = function (callback) {
-    this._loadMusics();
-    if (main.useCompress) {
-        this._load_async(callback);
-    } else {
-        this._load_sync(callback);
-    }
-}
+  this._loadMusics();
+  if (main.useCompress) {
+    this._load_async(callback);
+  } else {
+    this._load_sync(callback);
+  }
+};
 
 loader.prototype._load_sync = function (callback) {
-    this._loadAnimates_sync();
-    this._loadSounds_sync();
-    core.loader._loadMaterials_sync(function () {
-        core.loader._loadExtraImages_sync(function () {
-            core.loader._loadAutotiles_sync(function () {
-                core.loader._loadTilesets_sync(callback);
-            })
-        })
+  this._loadAnimates_sync();
+  this._loadSounds_sync();
+  core.loader._loadMaterials_sync(function () {
+    core.loader._loadExtraImages_sync(function () {
+      core.loader._loadAutotiles_sync(function () {
+        core.loader._loadTilesets_sync(callback);
+      });
     });
-}
+  });
+};
 
 loader.prototype._load_async = function (callback) {
-    core.loader._setStartLoadTipText('正在加载资源文件...');
-    const all = {};
+  core.loader._setStartLoadTipText("正在加载资源文件...");
+  const all = {};
 
-    const _makeOnProgress = function (name) {
-        if (!all[name]) all[name] = { loaded: 0, total: 0, finished: false };
-        return (loaded, total) => {
-            all[name].loaded = loaded;
-            all[name].total = total;
-            let allLoaded = 0, allTotal = 0;
-            for (const one of Object.values(all)) {
-                allLoaded += one.loaded;
-                allTotal += one.total;
-            }
-            console.log(allLoaded. allTotal);
-            if (allTotal > 0) {
-                if (allLoaded == allTotal) {
-                    core.loader._setStartLoadTipText("正在处理资源文件... 请稍候...");
-                } else {
-                    core.loader._setStartLoadTipText('正在加载资源文件... ' +
-                        core.formatSize(allLoaded) + " / " + core.formatSize(allTotal) +
-                        " (" + (allLoaded / allTotal * 100).toFixed(2) + "%)");
-                }
-                core.loader._setStartProgressVal(allLoaded / allTotal * 100);
-            }
-        };
-    }
-    const _makeOnFinished = function (name) {
-        return () => {
-            setTimeout(() => {
-                all[name].finished = true;
-                for (var one in all) {
-                    if (!all[one].finished) return;
-                }
-                callback();
-            });
+  const _makeOnProgress = function (name) {
+    if (!all[name]) all[name] = { loaded: 0, total: 0, finished: false };
+    return (loaded, total) => {
+      all[name].loaded = loaded;
+      all[name].total = total;
+      let allLoaded = 0,
+        allTotal = 0;
+      for (const one of Object.values(all)) {
+        allLoaded += one.loaded;
+        allTotal += one.total;
+      }
+      console.log(allLoaded.allTotal);
+      if (allTotal > 0) {
+        if (allLoaded == allTotal) {
+          core.loader._setStartLoadTipText("正在处理资源文件... 请稍候...");
+        } else {
+          core.loader._setStartLoadTipText(
+            "正在加载资源文件... " +
+              core.formatSize(allLoaded) +
+              " / " +
+              core.formatSize(allTotal) +
+              " (" +
+              ((allLoaded / allTotal) * 100).toFixed(2) +
+              "%)"
+          );
         }
-    }
+        core.loader._setStartProgressVal((allLoaded / allTotal) * 100);
+      }
+    };
+  };
+  const _makeOnFinished = function (name) {
+    return () => {
+      setTimeout(() => {
+        all[name].finished = true;
+        for (var one in all) {
+          if (!all[one].finished) return;
+        }
+        callback();
+      });
+    };
+  };
 
-    if (main.splitChunkMap) {
-        this._loadAnimates_chunked(main.splitChunkMap.animates, _makeOnProgress, _makeOnFinished);
-        this._loadSounds_chunked(main.splitChunkMap.sounds, _makeOnProgress, _makeOnFinished);
-        this._loadMaterials_chunked(main.splitChunkMap.materials, _makeOnProgress, _makeOnFinished);
-        this._loadExtraImages_chunked(main.splitChunkMap.images, _makeOnProgress, _makeOnFinished);
-        this._loadAutotiles_chunked(main.splitChunkMap.autotiles, _makeOnProgress, _makeOnFinished);
-        this._loadTilesets_chunked(main.splitChunkMap.tilesets, _makeOnProgress, _makeOnFinished);
-    } else {
-        this._loadAnimates_async(_makeOnProgress('animates'), _makeOnFinished('animates'));
-        this._loadSounds_async(_makeOnProgress('sounds'), _makeOnFinished('sounds'));
-        this._loadMaterials_async(_makeOnProgress('materials'), _makeOnFinished('materials'));
-        this._loadExtraImages_async(_makeOnProgress('images'), _makeOnFinished('images'));
-        this._loadAutotiles_async(_makeOnProgress('autotiles'), _makeOnFinished('autotiles'));
-        this._loadTilesets_async(_makeOnProgress('tilesets'), _makeOnFinished('tilesets'));
-    }
-}
+  if (main.splitChunkMap) {
+    this._loadAnimates_chunked(
+      main.splitChunkMap.animates,
+      _makeOnProgress,
+      _makeOnFinished
+    );
+    this._loadSounds_chunked(
+      main.splitChunkMap.sounds,
+      _makeOnProgress,
+      _makeOnFinished
+    );
+    this._loadMaterials_chunked(
+      main.splitChunkMap.materials,
+      _makeOnProgress,
+      _makeOnFinished
+    );
+    this._loadExtraImages_chunked(
+      main.splitChunkMap.images,
+      _makeOnProgress,
+      _makeOnFinished
+    );
+    this._loadAutotiles_chunked(
+      main.splitChunkMap.autotiles,
+      _makeOnProgress,
+      _makeOnFinished
+    );
+    this._loadTilesets_chunked(
+      main.splitChunkMap.tilesets,
+      _makeOnProgress,
+      _makeOnFinished
+    );
+  } else {
+    this._loadAnimates_async(
+      _makeOnProgress("animates"),
+      _makeOnFinished("animates")
+    );
+    this._loadSounds_async(
+      _makeOnProgress("sounds"),
+      _makeOnFinished("sounds")
+    );
+    this._loadMaterials_async(
+      _makeOnProgress("materials"),
+      _makeOnFinished("materials")
+    );
+    this._loadExtraImages_async(
+      _makeOnProgress("images"),
+      _makeOnFinished("images")
+    );
+    this._loadAutotiles_async(
+      _makeOnProgress("autotiles"),
+      _makeOnFinished("autotiles")
+    );
+    this._loadTilesets_async(
+      _makeOnProgress("tilesets"),
+      _makeOnFinished("tilesets")
+    );
+  }
+};
 
 // ----- 加载资源文件 ------ //
 
 loader.prototype._loadMaterials_sync = function (callback) {
-    this._setStartLoadTipText("正在加载资源文件...");
-    this.loadImages("materials", core.materials, core.material.images, function () {
-        core.loader._loadMaterials_afterLoad();
-        callback();
-    });
-}
+  this._setStartLoadTipText("正在加载资源文件...");
+  this.loadImages(
+    "materials",
+    core.materials,
+    core.material.images,
+    function () {
+      core.loader._loadMaterials_afterLoad();
+      callback();
+    }
+  );
+};
 
 loader.prototype._loadMaterials_async = function (onprogress, onfinished) {
-    this.loadImagesFromZip('project/materials/materials.h5data', core.materials, core.material.images, onprogress, function () {
-        core.loader._loadMaterials_afterLoad();
-        onfinished();
-    });
-}
+  this.loadImagesFromZip(
+    "project/materials/materials.h5data",
+    core.materials,
+    core.material.images,
+    onprogress,
+    function () {
+      core.loader._loadMaterials_afterLoad();
+      onfinished();
+    }
+  );
+};
 
-loader.prototype._loadMaterials_chunked = async function (chunks, makeOnProgress, makeOnFinished) {
-    await this._loadImagesFromChunks(chunks, core.materials, core.material.images, makeOnProgress, makeOnFinished);
-    core.loader._loadMaterials_afterLoad();
-}
+loader.prototype._loadMaterials_chunked = async function (
+  chunks,
+  makeOnProgress,
+  makeOnFinished
+) {
+  await this._loadImagesFromChunks(
+    chunks,
+    core.materials,
+    core.material.images,
+    makeOnProgress,
+    makeOnFinished
+  );
+  core.loader._loadMaterials_afterLoad();
+};
 
 loader.prototype._loadMaterials_afterLoad = function () {
-    const images = core.splitImage(core.material.images['icons']);
-    for (let key in core.statusBar.icons) {
-        if (typeof core.statusBar.icons[key] == 'number') {
-            core.statusBar.icons[key] = images[core.statusBar.icons[key]];
-            if (core.statusBar.image[key] != null)
-                core.statusBar.image[key].src = core.statusBar.icons[key].src;
-        }
+  const images = core.splitImage(core.material.images["icons"]);
+  for (let key in core.statusBar.icons) {
+    if (typeof core.statusBar.icons[key] == "number") {
+      core.statusBar.icons[key] = images[core.statusBar.icons[key]];
+      if (core.statusBar.image[key] != null)
+        core.statusBar.image[key].src = core.statusBar.icons[key].src;
     }
-}
+  }
+};
 
 // ------ 加载使用的图片 ------ //
 
 loader.prototype._loadExtraImages_sync = function (callback) {
-    core.material.images.images = {};
-    this._setStartLoadTipText("正在加载图片文件...");
-    core.loadImages("images", core.images, core.material.images.images, callback);
-}
+  core.material.images.images = {};
+  this._setStartLoadTipText("正在加载图片文件...");
+  core.loadImages("images", core.images, core.material.images.images, callback);
+};
 
 loader.prototype._loadExtraImages_async = function (onprogress, onfinished) {
-    core.material.images.images = {};
-    
-    // Check .gif
-    const gifs = images.filter(function (name) {
-        return name.toLowerCase().endsWith('.gif');
-    });
-    // gif没有被压缩在zip中,延迟加载...
-    this._loadExtraImages_loadLazy(gifs);
+  core.material.images.images = {};
 
-    images = images.filter((name) => !name.toLowerCase().endsWith('.gif'));
+  // Check .gif
+  const gifs = images.filter(function (name) {
+    return name.toLowerCase().endsWith(".gif");
+  });
+  // gif没有被压缩在zip中,延迟加载...
+  this._loadExtraImages_loadLazy(gifs);
 
-    this.loadImagesFromZip('project/images/images.h5data', images, core.material.images.images, onprogress, onfinished);
-}
+  images = images.filter((name) => !name.toLowerCase().endsWith(".gif"));
 
-loader.prototype._loadExtraImages_chunked = function (chunks, makeOnProgress, makeOnFinished) {
-    core.material.images.images = {};
-    let images = core.images;
+  this.loadImagesFromZip(
+    "project/images/images.h5data",
+    images,
+    core.material.images.images,
+    onprogress,
+    onfinished
+  );
+};
 
-    // Check .gif
-    const gifs = images.filter(function (name) {
-        return name.toLowerCase().endsWith('.gif');
-    });
-    // gif没有被压缩在zip中,延迟加载...
-    this._loadExtraImages_loadLazy(gifs);
+loader.prototype._loadExtraImages_chunked = function (
+  chunks,
+  makeOnProgress,
+  makeOnFinished
+) {
+  core.material.images.images = {};
+  let images = core.images;
 
-    images = images.filter((name) => !name.toLowerCase().endsWith('.gif'));
+  // Check .gif
+  const gifs = images.filter(function (name) {
+    return name.toLowerCase().endsWith(".gif");
+  });
+  // gif没有被压缩在zip中,延迟加载...
+  this._loadExtraImages_loadLazy(gifs);
 
-    this._loadImagesFromChunks(chunks, images, core.material.images.images, makeOnProgress, makeOnFinished);
-}
+  images = images.filter((name) => !name.toLowerCase().endsWith(".gif"));
+
+  this._loadImagesFromChunks(
+    chunks,
+    images,
+    core.material.images.images,
+    makeOnProgress,
+    makeOnFinished
+  );
+};
 
 loader.prototype._loadExtraImages_loadLazy = function (list) {
-    list.forEach(function (gif) {
-        this.loadImage("images", gif, (id, image) => {
-            if (image != null) {
-                core.material.images.images[gif] = image;
-            }
-        });
-    }, this);
-}
+  list.forEach(function (gif) {
+    this.loadImage("images", gif, (id, image) => {
+      if (image != null) {
+        core.material.images.images[gif] = image;
+      }
+    });
+  }, this);
+};
 
 // ------ 加载自动元件 ------ //
 
 loader.prototype._loadAutotiles_sync = function (callback) {
-    core.material.images.autotile = {};
-    var keys = Object.keys(core.material.icons.autotile);
-    var autotiles = {};
+  core.material.images.autotile = {};
+  var keys = Object.keys(core.material.icons.autotile);
+  var autotiles = {};
 
-    this._setStartLoadTipText("正在加载自动元件...");
-    this.loadImages("autotiles", keys, autotiles, function () {
-        core.loader._loadAutotiles_afterLoad(keys, autotiles);
-        callback();
-    });
-}
+  this._setStartLoadTipText("正在加载自动元件...");
+  this.loadImages("autotiles", keys, autotiles, function () {
+    core.loader._loadAutotiles_afterLoad(keys, autotiles);
+    callback();
+  });
+};
 
 loader.prototype._loadAutotiles_async = function (onprogress, onfinished) {
-    core.material.images.autotile = {};
-    var keys = Object.keys(core.material.icons.autotile);
-    var autotiles = {};
+  core.material.images.autotile = {};
+  var keys = Object.keys(core.material.icons.autotile);
+  var autotiles = {};
 
-    this.loadImagesFromZip('project/autotiles/autotiles.h5data', keys, autotiles, onprogress, function () {
-        core.loader._loadAutotiles_afterLoad(keys, autotiles);
-        onfinished();
-    });
-}
+  this.loadImagesFromZip(
+    "project/autotiles/autotiles.h5data",
+    keys,
+    autotiles,
+    onprogress,
+    function () {
+      core.loader._loadAutotiles_afterLoad(keys, autotiles);
+      onfinished();
+    }
+  );
+};
 
-loader.prototype._loadAutotiles_chunked = async function (chunks, makeOnProgress, makeOnFinished) {
-    core.material.images.autotile = {};
-    const keys = Object.keys(core.material.icons.autotile);
-    const autotiles = {};
+loader.prototype._loadAutotiles_chunked = async function (
+  chunks,
+  makeOnProgress,
+  makeOnFinished
+) {
+  core.material.images.autotile = {};
+  const keys = Object.keys(core.material.icons.autotile);
+  const autotiles = {};
 
-    await this._loadImagesFromChunks(chunks, keys, autotiles, makeOnProgress, makeOnFinished);
-    core.loader._loadAutotiles_afterLoad(keys, autotiles);
-}
+  await this._loadImagesFromChunks(
+    chunks,
+    keys,
+    autotiles,
+    makeOnProgress,
+    makeOnFinished
+  );
+  core.loader._loadAutotiles_afterLoad(keys, autotiles);
+};
 
 loader.prototype._loadAutotiles_afterLoad = function (keys, autotiles) {
-    // autotile需要保证顺序
-    keys.forEach(function (v) {
-        core.material.images.autotile[v] = autotiles[v];
-    });
+  // autotile需要保证顺序
+  keys.forEach(function (v) {
+    core.material.images.autotile[v] = autotiles[v];
+  });
 
-    setTimeout(function () {
-        core.maps._makeAutotileEdges();
-    });
-
-}
+  setTimeout(function () {
+    core.maps._makeAutotileEdges();
+  });
+};
 
 // ------ 加载额外素材 ------ //
 
 loader.prototype._loadTilesets_sync = function (callback) {
-    core.material.images.tilesets = {};
-    this._setStartLoadTipText("正在加载额外素材...");
-    this.loadImages("tilesets", core.tilesets, core.material.images.tilesets, function () {
-        core.loader._loadTilesets_afterLoad();
-        callback();
-    });
-}
+  core.material.images.tilesets = {};
+  this._setStartLoadTipText("正在加载额外素材...");
+  this.loadImages(
+    "tilesets",
+    core.tilesets,
+    core.material.images.tilesets,
+    function () {
+      core.loader._loadTilesets_afterLoad();
+      callback();
+    }
+  );
+};
 
 loader.prototype._loadTilesets_async = function (onprogress, onfinished) {
-    core.material.images.tilesets = {};
-    this.loadImagesFromZip('project/tilesets/tilesets.h5data', core.tilesets, core.material.images.tilesets, onprogress, function () {
-        core.loader._loadTilesets_afterLoad();
-        onfinished();
-    });
-}
+  core.material.images.tilesets = {};
+  this.loadImagesFromZip(
+    "project/tilesets/tilesets.h5data",
+    core.tilesets,
+    core.material.images.tilesets,
+    onprogress,
+    function () {
+      core.loader._loadTilesets_afterLoad();
+      onfinished();
+    }
+  );
+};
 
-loader.prototype._loadTilesets_chunked = async function (chunks, makeOnProgress, makeOnFinished) {
-    core.material.images.tilesets = {};
-    await this._loadImagesFromChunks(chunks, core.tilesets, core.material.images.tilesets, makeOnProgress, makeOnFinished);
-    core.loader._loadTilesets_afterLoad();
-}
+loader.prototype._loadTilesets_chunked = async function (
+  chunks,
+  makeOnProgress,
+  makeOnFinished
+) {
+  core.material.images.tilesets = {};
+  await this._loadImagesFromChunks(
+    chunks,
+    core.tilesets,
+    core.material.images.tilesets,
+    makeOnProgress,
+    makeOnFinished
+  );
+  core.loader._loadTilesets_afterLoad();
+};
 
 loader.prototype._loadTilesets_afterLoad = function () {
-    // 检查宽高是32倍数,如果出错在控制台报错
-    for (var imgName in core.material.images.tilesets) {
-        var img = core.material.images.tilesets[imgName];
-        if (img.width % 32 != 0 || img.height % 32 != 0) {
-            console.warn("警告!" + imgName + "的宽或高不是32的倍数!");
-        }
-        if (img.width * img.height > 32 * 32 * 3000) {
-            console.warn("警告!" + imgName + "上的图块素材个数大于3000!");
-        }
+  // 检查宽高是32倍数,如果出错在控制台报错
+  for (var imgName in core.material.images.tilesets) {
+    var img = core.material.images.tilesets[imgName];
+    if (img.width % 32 != 0 || img.height % 32 != 0) {
+      console.warn("警告!" + imgName + "的宽或高不是32的倍数!");
     }
-}
+    if (img.width * img.height > 32 * 32 * 3000) {
+      console.warn("警告!" + imgName + "上的图块素材个数大于3000!");
+    }
+  }
+};
 
 // ------ 实际加载一系列图片 ------ //
 
 loader.prototype.loadImages = function (dir, names, toSave, callback) {
-    if (!names || names.length == 0) {
-        if (callback) callback();
+  if (!names || names.length == 0) {
+    if (callback) callback();
+    return;
+  }
+  var items = 0;
+  for (var i = 0; i < names.length; i++) {
+    this.loadImage(dir, names[i], function (id, image) {
+      core.loader._setStartLoadTipText("正在加载图片 " + id + "...");
+      if (toSave[id] !== undefined) {
+        if (image != null) toSave[id] = image;
         return;
-    }
-    var items = 0;
-    for (var i = 0; i < names.length; i++) {
-        this.loadImage(dir, names[i], function (id, image) {
-            core.loader._setStartLoadTipText('正在加载图片 ' + id + "...");
-            if (toSave[id] !== undefined) {
-                if (image != null)
-                    toSave[id] = image;
-                return;
-            }
-            toSave[id] = image;
-            items++;
-            core.loader._setStartProgressVal(items * (100 / names.length));
-            if (items == names.length) {
-                if (callback) callback();
-            }
-        })
-    }
-}
+      }
+      toSave[id] = image;
+      items++;
+      core.loader._setStartProgressVal(items * (100 / names.length));
+      if (items == names.length) {
+        if (callback) callback();
+      }
+    });
+  }
+};
 
 loader.prototype.loadImage = function (dir, imgName, callback) {
-    try {
-        var name = imgName;
-        if (name.indexOf(".") < 0)
-            name = name + ".png";
-        var image = new Image();
-        image.onload = function () {
-            image.setAttribute('_width', image.width);
-            image.setAttribute('_height', image.height);
-            callback(imgName, image);
-        }
-        image.onerror = function () {
-            callback(imgName, null);
-        }
-        image.src = 'project/' + dir + '/' + name + "?v=" + main.version;
-        if (name.endsWith('.gif'))
-            callback(imgName, null);
-    }
-    catch (e) {
-        console.error(e);
-    }
-}
+  try {
+    var name = imgName;
+    if (name.indexOf(".") < 0) name = name + ".png";
+    var image = new Image();
+    image.onload = function () {
+      image.setAttribute("_width", image.width);
+      image.setAttribute("_height", image.height);
+      callback(imgName, image);
+    };
+    image.onerror = function () {
+      callback(imgName, null);
+    };
+    image.src = "project/" + dir + "/" + name + "?v=" + main.version;
+    if (name.endsWith(".gif")) callback(imgName, null);
+  } catch (e) {
+    console.error(e);
+  }
+};
 
 // ------ 从zip中加载一系列图片 ------ //
 
-loader.prototype.loadImagesFromZip = function (url, names, toSave, onprogress, onfinished) {
-    if (!names || names.length == 0) {
-        if (onfinished) onfinished();
-        return;
-    }
+loader.prototype.loadImagesFromZip = function (
+  url,
+  names,
+  toSave,
+  onprogress,
+  onfinished
+) {
+  if (!names || names.length == 0) {
+    if (onfinished) onfinished();
+    return;
+  }
 
-    core.unzip(url + "?v=" + main.version, function (data) {
-        var cnt = 1;
-        names.forEach(function (name) {
-            var imgName = name;
-            if (imgName.indexOf('.') < 0) imgName += '.png';
-            if (imgName in data) {
-                var img = new Image();
-                var url = URL.createObjectURL(data[imgName]);
-                cnt++;
-                img.onload = function () {
-                    cnt--;
-                    URL.revokeObjectURL(url);
-                    img.setAttribute('_width', img.width);
-                    img.setAttribute('_height', img.height);
-                    if (cnt == 0 && onfinished) onfinished();
-                }
-                img.src = url;
-                toSave[name] = img;
-            }
-        });
-        cnt--;
-        if (cnt == 0 && onfinished) onfinished();
-    }, null, false, onprogress);
-}
+  core.unzip(
+    url + "?v=" + main.version,
+    function (data) {
+      var cnt = 1;
+      names.forEach(function (name) {
+        var imgName = name;
+        if (imgName.indexOf(".") < 0) imgName += ".png";
+        if (imgName in data) {
+          var img = new Image();
+          var url = URL.createObjectURL(data[imgName]);
+          cnt++;
+          img.onload = function () {
+            cnt--;
+            URL.revokeObjectURL(url);
+            img.setAttribute("_width", img.width);
+            img.setAttribute("_height", img.height);
+            if (cnt == 0 && onfinished) onfinished();
+          };
+          img.src = url;
+          toSave[name] = img;
+        }
+      });
+      cnt--;
+      if (cnt == 0 && onfinished) onfinished();
+    },
+    null,
+    false,
+    onprogress
+  );
+};
 
-loader.prototype._loadImagesFromChunks = async function (chunks, names, toSave, makeOnProgress, makeOnFinished) {
-    await Promise.all(chunks.map((chunk) => {
-        const onfinished = makeOnFinished(chunk);
-        const onprogress = makeOnProgress(chunk);
-        return new Promise((resolve) => {
-            this.loadImagesFromZip(chunk, names, toSave, onprogress, () => {
-                onfinished();
-                resolve();     
-            });
+loader.prototype._loadImagesFromChunks = async function (
+  chunks,
+  names,
+  toSave,
+  makeOnProgress,
+  makeOnFinished
+) {
+  await Promise.all(
+    chunks.map((chunk) => {
+      const onfinished = makeOnFinished(chunk);
+      const onprogress = makeOnProgress(chunk);
+      return new Promise((resolve) => {
+        this.loadImagesFromZip(chunk, names, toSave, onprogress, () => {
+          onfinished();
+          resolve();
         });
-    }));
-}
+      });
+    })
+  );
+};
 
 // ------ 加载动画文件 ------ //
 
 loader.prototype._loadAnimates_sync = function () {
-    this._setStartLoadTipText("正在加载动画文件...");
+  this._setStartLoadTipText("正在加载动画文件...");
 
-    if (main.supportBunch) {
-        if (core.animates.length > 0) {
-            core.http('GET', '__all_animates__?v=' + main.version + '&id=' + core.animates.join(','), null, function (content) {
-                var u = content.split('@@@~~~###~~~@@@');
-                for (var i = 0; i < core.animates.length; ++i) {
-                    if (u[i] != '') {
-                        core.material.animates[core.animates[i]] = core.loader._loadAnimate(u[i]);
-                    } else {
-                        console.error('无法找到动画文件' + core.animates[i] + '!');
-                    }
-                }
-            }, "text/plain; charset=x-user-defined");
-        }
-        return;
+  if (main.supportBunch) {
+    if (core.animates.length > 0) {
+      core.http(
+        "GET",
+        "__all_animates__?v=" + main.version + "&id=" + core.animates.join(","),
+        null,
+        function (content) {
+          var u = content.split("@@@~~~###~~~@@@");
+          for (var i = 0; i < core.animates.length; ++i) {
+            if (u[i] != "") {
+              core.material.animates[core.animates[i]] =
+                core.loader._loadAnimate(u[i]);
+            } else {
+              console.error("无法找到动画文件" + core.animates[i] + "!");
+            }
+          }
+        },
+        "text/plain; charset=x-user-defined"
+      );
     }
+    return;
+  }
 
-    core.animates.forEach(function (t) {
-        core.http('GET', 'project/animates/' + t + ".animate?v=" + main.version, null, function (content) {
-            core.material.animates[t] = core.loader._loadAnimate(content);
-        }, function (e) {
-            console.error(e);
-            core.material.animates[t] = null;
-        }, "text/plain; charset=x-user-defined")
-    });
-}
+  core.animates.forEach(function (t) {
+    core.http(
+      "GET",
+      "project/animates/" + t + ".animate?v=" + main.version,
+      null,
+      function (content) {
+        core.material.animates[t] = core.loader._loadAnimate(content);
+      },
+      function (e) {
+        console.error(e);
+        core.material.animates[t] = null;
+      },
+      "text/plain; charset=x-user-defined"
+    );
+  });
+};
 
 loader.prototype._loadAnimates_async = function (onprogress, onfinished) {
-    this._loadFileFromZip('project/animates/animates.h5data', this._saveAnimate, true, onprogress, onfinished);
-}
+  this._loadFileFromZip(
+    "project/animates/animates.h5data",
+    this._saveAnimate,
+    true,
+    onprogress,
+    onfinished
+  );
+};
 
-loader.prototype._loadAnimates_chunked = function (chunks, makeOnProgress, makeOnFinished) {
-    this._loadFileFromChunks(chunks, this._saveAnimate, true, makeOnProgress, makeOnFinished);
-}
+loader.prototype._loadAnimates_chunked = function (
+  chunks,
+  makeOnProgress,
+  makeOnFinished
+) {
+  this._loadFileFromChunks(
+    chunks,
+    this._saveAnimate,
+    true,
+    makeOnProgress,
+    makeOnFinished
+  );
+};
 
 loader.prototype._saveAnimate = function (animates, onfinished) {
-    for (var name in animates) {
-        if (name.endsWith(".animate")) {
-            var t = name.substring(0, name.length - 8);
-            if (core.animates.indexOf(t) >= 0)
-                core.material.animates[t] = core.loader._loadAnimate(animates[name]);
-        }
+  for (var name in animates) {
+    if (name.endsWith(".animate")) {
+      var t = name.substring(0, name.length - 8);
+      if (core.animates.indexOf(t) >= 0)
+        core.material.animates[t] = core.loader._loadAnimate(animates[name]);
     }
-    onfinished();
-}
+  }
+  onfinished();
+};
 
 loader.prototype._loadAnimate = function (content) {
-    try {
-        content = JSON.parse(content);
-        var data = {};
-        data.ratio = content.ratio;
-        data.se = content.se;
-        data.pitch = content.pitch;
-        data.images = [];
-        content.bitmaps.forEach(function (t2) {
-            if (!t2) {
-                data.images.push(null);
-            }
-            else {
-                try {
-                    var image = new Image();
-                    image.src = t2;
-                    data.images.push(image);
-                } catch (e) {
-                    console.error(e);
-                    data.images.push(null);
-                }
-            }
-        })
-        data.frame = content.frame_max;
-        data.frames = [];
-        content.frames.forEach(function (t2) {
-            var info = [];
-            t2.forEach(function (t3) {
-                info.push({
-                    'index': t3[0],
-                    'x': t3[1],
-                    'y': t3[2],
-                    'zoom': t3[3],
-                    'opacity': t3[4],
-                    'mirror': t3[5] || 0,
-                    'angle': t3[6] || 0,
-                })
-            })
-            data.frames.push(info);
+  try {
+    content = JSON.parse(content);
+    var data = {};
+    data.ratio = content.ratio;
+    data.se = content.se;
+    data.pitch = content.pitch;
+    data.images = [];
+    content.bitmaps.forEach(function (t2) {
+      if (!t2) {
+        data.images.push(null);
+      } else {
+        try {
+          var image = new Image();
+          image.src = t2;
+          data.images.push(image);
+        } catch (e) {
+          console.error(e);
+          data.images.push(null);
+        }
+      }
+    });
+    data.frame = content.frame_max;
+    data.frames = [];
+    content.frames.forEach(function (t2) {
+      var info = [];
+      t2.forEach(function (t3) {
+        info.push({
+          index: t3[0],
+          x: t3[1],
+          y: t3[2],
+          zoom: t3[3],
+          opacity: t3[4],
+          mirror: t3[5] || 0,
+          angle: t3[6] || 0,
         });
-        return data;
-    }
-    catch (e) {
-        console.error(e);
-        return null;
-    }
-}
+      });
+      data.frames.push(info);
+    });
+    return data;
+  } catch (e) {
+    console.error(e);
+    return null;
+  }
+};
 
 // ------ 加载音乐和音效 ------ //
 
 loader.prototype._loadMusics = function () {
-    core.bgms.forEach(function (t) {
-        core.loader.loadOneMusic(t);
-    });
-    // 直接开始播放
-    core.playBgm(main.startBgm);
-}
+  core.bgms.forEach(function (t) {
+    core.loader.loadOneMusic(t);
+  });
+  // 直接开始播放
+  core.playBgm(main.startBgm);
+};
 
 loader.prototype._loadSounds_sync = function () {
-    this._setStartLoadTipText("正在加载音效文件...");
-    core.sounds.forEach(function (t) {
-        core.loader.loadOneSound(t);
-    });
-}
+  this._setStartLoadTipText("正在加载音效文件...");
+  core.sounds.forEach(function (t) {
+    core.loader.loadOneSound(t);
+  });
+};
 
 loader.prototype._loadSounds_async = function (onprogress, onfinished) {
-    this._loadFileFromZip('project/sounds/sounds.h5data', this._saveSounds, false, onprogress, onfinished);
-}
+  this._loadFileFromZip(
+    "project/sounds/sounds.h5data",
+    this._saveSounds,
+    false,
+    onprogress,
+    onfinished
+  );
+};
 
-loader.prototype._loadSounds_chunked = function (chunks, makeOnProgress, makeOnFinished) {
-    this._loadFileFromChunks(chunks, this._saveSounds, false, makeOnProgress, makeOnFinished);
-}
+loader.prototype._loadSounds_chunked = function (
+  chunks,
+  makeOnProgress,
+  makeOnFinished
+) {
+  this._loadFileFromChunks(
+    chunks,
+    this._saveSounds,
+    false,
+    makeOnProgress,
+    makeOnFinished
+  );
+};
 
 loader.prototype._saveSounds = function (data, onfinished) {
-    // 延迟解析
-    setTimeout(function () {
-        for (var name in data) {
-            if (core.sounds.indexOf(name) >= 0) {
-                core.loader._loadOneSound_decodeData(name, data[name]);
-            }
-        }
-        onfinished();
-    });
-}
+  // 延迟解析
+  setTimeout(function () {
+    for (var name in data) {
+      if (core.sounds.indexOf(name) >= 0) {
+        core.loader._loadOneSound_decodeData(name, data[name]);
+      }
+    }
+    onfinished();
+  });
+};
 
 loader.prototype.loadOneMusic = function (name) {
-    var music = new Audio();
-    music.preload = 'none';
-    if (main.bgmRemote) music.src = main.bgmRemoteRoot + core.firstData.name + '/' + name;
-    else music.src = 'project/bgms/' + name;
-    music.loop = 'loop';
-    core.material.bgms[name] = music;
-}
+  /* var music = new Audio();
+  music.preload = "none";
+  if (main.bgmRemote)
+    music.src = main.bgmRemoteRoot + core.firstData.name + "/" + name;
+  else music.src = "project/bgms/" + name;
+  music.loop = "loop";
+  core.material.bgms[name] = music;*/
+};
 
 loader.prototype.loadOneSound = function (name) {
-    core.http('GET', 'project/sounds/' + name + "?v=" + main.version, null, function (data) {
-        core.loader._loadOneSound_decodeData(name, data);
-    }, function (e) {
-        console.error(e);
-        core.material.sounds[name] = null;
-    }, null, 'arraybuffer');
-}
+  core.http(
+    "GET",
+    "project/sounds/" + name + "?v=" + main.version,
+    null,
+    function (data) {
+      core.loader._loadOneSound_decodeData(name, data);
+    },
+    function (e) {
+      console.error(e);
+      core.material.sounds[name] = null;
+    },
+    null,
+    "arraybuffer"
+  );
+};
 
 loader.prototype._loadOneSound_decodeData = function (name, data) {
-    if (data instanceof Blob) {
-        var blobReader = new zip.BlobReader(data);
-        blobReader.init(function () {
-            blobReader.readUint8Array(0, blobReader.size, function (uint8) {
-                core.loader._loadOneSound_decodeData(name, uint8.buffer);
-            })
-        });
-        return;
-    }
-    try {
-        core.musicStatus.audioContext.decodeAudioData(data, function (buffer) {
-            core.material.sounds[name] = buffer;
-        }, function (e) {
-            console.error(e);
-            core.material.sounds[name] = null;
-        })
-    }
-    catch (e) {
+  if (data instanceof Blob) {
+    var blobReader = new zip.BlobReader(data);
+    blobReader.init(function () {
+      blobReader.readUint8Array(0, blobReader.size, function (uint8) {
+        core.loader._loadOneSound_decodeData(name, uint8.buffer);
+      });
+    });
+    return;
+  }
+  try {
+    core.musicStatus.audioContext.decodeAudioData(
+      data,
+      function (buffer) {
+        core.material.sounds[name] = buffer;
+      },
+      function (e) {
         console.error(e);
         core.material.sounds[name] = null;
-    }
-}
+      }
+    );
+  } catch (e) {
+    console.error(e);
+    core.material.sounds[name] = null;
+  }
+};
 
 loader.prototype.loadBgm = function (name) {
-    name = core.getMappedName(name);
-    if (!core.material.bgms[name]) return;
-    // 如果没开启音乐,则不预加载
-    if (!core.musicStatus.bgmStatus) return;
-    // 是否已经预加载过
-    var index = core.musicStatus.cachedBgms.indexOf(name);
-    if (index >= 0) {
-        core.musicStatus.cachedBgms.splice(index, 1);
+  name = core.getMappedName(name);
+  if (!core.material.bgms[name]) return;
+  // 如果没开启音乐,则不预加载
+  if (!core.musicStatus.bgmStatus) return;
+  // 是否已经预加载过
+  var index = core.musicStatus.cachedBgms.indexOf(name);
+  if (index >= 0) {
+    core.musicStatus.cachedBgms.splice(index, 1);
+  } else {
+    // 预加载BGM
+    this._preloadBgm(core.material.bgms[name]);
+    // core.material.bgms[name].load();
+    // 清理尾巴
+    if (core.musicStatus.cachedBgms.length == core.musicStatus.cachedBgmCount) {
+      this.freeBgm(core.musicStatus.cachedBgms.pop());
     }
-    else {
-        // 预加载BGM
-        this._preloadBgm(core.material.bgms[name]);
-        // core.material.bgms[name].load();
-        // 清理尾巴
-        if (core.musicStatus.cachedBgms.length == core.musicStatus.cachedBgmCount) {
-            this.freeBgm(core.musicStatus.cachedBgms.pop());
-        }
-    }
-    // 移动到缓存最前方
-    core.musicStatus.cachedBgms.unshift(name);
-}
+  }
+  // 移动到缓存最前方
+  core.musicStatus.cachedBgms.unshift(name);
+};
 
 loader.prototype._preloadBgm = function (bgm) {
-    bgm.volume = 0;
-    bgm.play();
-}
+  bgm.volume = 0;
+  bgm.play();
+};
 
 loader.prototype.freeBgm = function (name) {
-    name = core.getMappedName(name);
-    if (!core.material.bgms[name]) return;
-    // 从cachedBgms中删除
-    core.musicStatus.cachedBgms = core.musicStatus.cachedBgms.filter(function (t) {
-        return t != name;
-    });
-    // 清掉缓存
-    core.material.bgms[name].removeAttribute("src");
-    core.material.bgms[name].load();
-    core.material.bgms[name] = null;
-    if (name == core.musicStatus.playingBgm) {
-        core.musicStatus.playingBgm = null;
-    }
-    // 三秒后重新加载
-    setTimeout(function () {
-        core.loader.loadOneMusic(name);
-    }, 3000);
-}
+  name = core.getMappedName(name);
+  if (!core.material.bgms[name]) return;
+  // 从cachedBgms中删除
+  core.musicStatus.cachedBgms = core.musicStatus.cachedBgms.filter(function (
+    t
+  ) {
+    return t != name;
+  });
+  // 清掉缓存
+  core.material.bgms[name].removeAttribute("src");
+  core.material.bgms[name].load();
+  core.material.bgms[name] = null;
+  if (name == core.musicStatus.playingBgm) {
+    core.musicStatus.playingBgm = null;
+  }
+  // 三秒后重新加载
+  setTimeout(function () {
+    core.loader.loadOneMusic(name);
+  }, 3000);
+};
 
-loader.prototype._loadFileFromZip = function (url, save, convertToText, onprogress, onfinished) {
-    core.unzip(url + '?v=' + main.version, function (data) {
-        save(data, onfinished);
-    }, null, convertToText, onprogress);
-}
+loader.prototype._loadFileFromZip = function (
+  url,
+  save,
+  convertToText,
+  onprogress,
+  onfinished
+) {
+  core.unzip(
+    url + "?v=" + main.version,
+    function (data) {
+      save(data, onfinished);
+    },
+    null,
+    convertToText,
+    onprogress
+  );
+};
 
-loader.prototype._loadFileFromChunks = async function (chunks, save, convertToText, makeOnProgress, makeOnFinished) {
-    await Promise.all(chunks.map((chunk) => {
-        const onfinished = makeOnFinished(chunk);
-        const onprogress = makeOnProgress(chunk);
-        return new Promise((resolve) => {
-            this._loadFileFromZip(chunk, save, convertToText, onprogress, () => {
-                onfinished();
-                resolve();
-            });
+loader.prototype._loadFileFromChunks = async function (
+  chunks,
+  save,
+  convertToText,
+  makeOnProgress,
+  makeOnFinished
+) {
+  await Promise.all(
+    chunks.map((chunk) => {
+      const onfinished = makeOnFinished(chunk);
+      const onprogress = makeOnProgress(chunk);
+      return new Promise((resolve) => {
+        this._loadFileFromZip(chunk, save, convertToText, onprogress, () => {
+          onfinished();
+          resolve();
         });
-    }));
-}
+      });
+    })
+  );
+};
diff --git a/libs/maps.js b/libs/maps.js
index 4fbbb8b..35e6b8d 100644
--- a/libs/maps.js
+++ b/libs/maps.js
@@ -2192,6 +2192,7 @@ maps.prototype.showBgFgMap = function (name, loc, floorId, callback) {
 
 ////// 隐藏前景/背景地图 //////
 maps.prototype.hideBgFgMap = function (name, loc, floorId, callback) {
+    this.removeGlobalAnimate(loc.x, loc.y, name)
     this._triggerBgFgMap('hide', name, loc, floorId, callback);
 }
 
@@ -2455,7 +2456,7 @@ maps.prototype.setBgFgBlock = function (name, number, x, y, floorId) {
         if (/^\d+$/.test(number)) number = parseInt(number);
         else number = core.getNumberById(number);
     }
-
+    this.removeGlobalAnimate(x, y, name)
     var values = core.getFlag('__' + name + 'v__', {});
     values[floorId] = (values[floorId] || []).filter(function (one) { return one[0] != x || one[1] != y });
     values[floorId].push([x, y, number]);
diff --git a/libs/utils.js b/libs/utils.js
index cd5e576..40fe703 100644
--- a/libs/utils.js
+++ b/libs/utils.js
@@ -1,4 +1,3 @@
-
 /*
 utils.js 工具类
 
@@ -6,1335 +5,1488 @@ utils.js 工具类
 
 "use strict";
 
-function utils () {
-    this._init();
-    this.scan = {
-        'up': { 'x': 0, 'y': -1 },
-        'left': { 'x': -1, 'y': 0 },
-        'down': { 'x': 0, 'y': 1 },
-        'right': { 'x': 1, 'y': 0 }
-    };
-    this.scan2 = {
-        'up': { 'x': 0, 'y': -1 },
-        'left': { 'x': -1, 'y': 0 },
-        'down': { 'x': 0, 'y': 1 },
-        'right': { 'x': 1, 'y': 0 },
-        'leftup': { 'x': -1, 'y': -1 },
-        'leftdown': { 'x': -1, 'y': 1 },
-        'rightup': { 'x': 1, 'y': -1 },
-        'rightdown': { 'x': 1, 'y': 1 }
-    };
+function utils() {
+  this._init();
+  this.scan = {
+    up: { x: 0, y: -1 },
+    left: { x: -1, y: 0 },
+    down: { x: 0, y: 1 },
+    right: { x: 1, y: 0 },
+  };
+  this.scan2 = {
+    up: { x: 0, y: -1 },
+    left: { x: -1, y: 0 },
+    down: { x: 0, y: 1 },
+    right: { x: 1, y: 0 },
+    leftup: { x: -1, y: -1 },
+    leftdown: { x: -1, y: 1 },
+    rightup: { x: 1, y: -1 },
+    rightdown: { x: 1, y: 1 },
+  };
 }
 
 utils.prototype._init = function () {
-    //
-}
+  //
+};
 
 ////// 将文字中的${和}(表达式)进行替换 //////
 utils.prototype.replaceText = function (text, prefix) {
-    if (typeof text != 'string') return text;
-    var index = text.indexOf("${");
-    if (index < 0) return text;
-    var cnt = 0, curr = index;
-    while (++curr < text.length) {
-        if (text.charAt(curr) == '{') cnt++;
-        if (text.charAt(curr) == '}') cnt--;
-        if (cnt == 0) break;
-    }
-    if (cnt != 0) return text;
-    var value = core.calValue(text.substring(index + 2, curr), prefix);
-    if (value == null) value = "";
-    return text.substring(0, index) + value + core.replaceText(text.substring(curr + 1), prefix);
-}
+  if (typeof text != "string") return text;
+  var index = text.indexOf("${");
+  if (index < 0) return text;
+  var cnt = 0,
+    curr = index;
+  while (++curr < text.length) {
+    if (text.charAt(curr) == "{") cnt++;
+    if (text.charAt(curr) == "}") cnt--;
+    if (cnt == 0) break;
+  }
+  if (cnt != 0) return text;
+  var value = core.calValue(text.substring(index + 2, curr), prefix);
+  if (value == null) value = "";
+  return (
+    text.substring(0, index) +
+    value +
+    core.replaceText(text.substring(curr + 1), prefix)
+  );
+};
 
 utils.prototype.replaceValue = function (value) {
-    if (typeof value == "string" && (value.indexOf(":") >= 0 || value.indexOf("flag:") >= 0 || value.indexOf('global:') >= 0)) {
-        if (value.indexOf('status:') >= 0)
-            value = value.replace(/status:([a-zA-Z0-9_]+)/g, "core.getStatus('$1')");
-        if (value.indexOf('buff:') >= 0)
-            value = value.replace(/buff:([a-zA-Z0-9_]+)/g, "core.getBuff('$1')");
-        if (value.indexOf('item:') >= 0)
-            value = value.replace(/item:([a-zA-Z0-9_]+)/g, "core.itemCount('$1')");
-        if (value.indexOf('flag:') >= 0 || value.indexOf('flag:') >= 0)
-            value = value.replace(/flag[::]([a-zA-Z0-9_\u4E00-\u9FCC\u3040-\u30FF\u2160-\u216B\u0391-\u03C9]+)/g, "core.getFlag('$1', 0)");
-        //if (value.indexOf('switch:' >= 0))
-        //    value = value.replace(/switch:([a-zA-Z0-9_]+)/g, "core.getFlag('" + (prefix || ":f@x@y") + "@$1', 0)");
-        if (value.indexOf('global:') >= 0 || value.indexOf('global:') >= 0)
-            value = value.replace(/global[::]([a-zA-Z0-9_\u4E00-\u9FCC\u3040-\u30FF\u2160-\u216B\u0391-\u03C9]+)/g, "core.getGlobal('$1', 0)");
-        if (value.indexOf('enemy:') >= 0)
-            value = value.replace(/enemy:([a-zA-Z0-9_]+)[\.:]([a-zA-Z0-9_]+)/g, "core.material.enemys['$1'].$2");
-        if (value.indexOf('blockId:') >= 0)
-            value = value.replace(/blockId:(\d+),(\d+)/g, "core.getBlockId($1, $2)");
-        if (value.indexOf('blockNumber:') >= 0)
-            value = value.replace(/blockNumber:(\d+),(\d+)/g, "core.getBlockNumber($1, $2)");
-        if (value.indexOf('blockCls:') >= 0)
-            value = value.replace(/blockCls:(\d+),(\d+)/g, "core.getBlockCls($1, $2)");
-        if (value.indexOf('equip:') >= 0)
-            value = value.replace(/equip:(\d)/g, "core.getEquip($1)");
-        if (value.indexOf('temp:') >= 0)
-            value = value.replace(/temp:([a-zA-Z0-9_]+)/g, "core.getFlag('@temp@$1', 0)");
-    }
-    return value;
-}
+  if (
+    typeof value == "string" &&
+    (value.indexOf(":") >= 0 ||
+      value.indexOf("flag:") >= 0 ||
+      value.indexOf("global:") >= 0)
+  ) {
+    if (value.indexOf("status:") >= 0)
+      value = value.replace(/status:([a-zA-Z0-9_]+)/g, "core.getStatus('$1')");
+    if (value.indexOf("buff:") >= 0)
+      value = value.replace(/buff:([a-zA-Z0-9_]+)/g, "core.getBuff('$1')");
+    if (value.indexOf("item:") >= 0)
+      value = value.replace(/item:([a-zA-Z0-9_]+)/g, "core.itemCount('$1')");
+    if (value.indexOf("flag:") >= 0 || value.indexOf("flag:") >= 0)
+      value = value.replace(
+        /flag[::]([a-zA-Z0-9_\u4E00-\u9FCC\u3040-\u30FF\u2160-\u216B\u0391-\u03C9]+)/g,
+        "core.getFlag('$1', 0)"
+      );
+    //if (value.indexOf('switch:' >= 0))
+    //    value = value.replace(/switch:([a-zA-Z0-9_]+)/g, "core.getFlag('" + (prefix || ":f@x@y") + "@$1', 0)");
+    if (value.indexOf("global:") >= 0 || value.indexOf("global:") >= 0)
+      value = value.replace(
+        /global[::]([a-zA-Z0-9_\u4E00-\u9FCC\u3040-\u30FF\u2160-\u216B\u0391-\u03C9]+)/g,
+        "core.getGlobal('$1', 0)"
+      );
+    if (value.indexOf("enemy:") >= 0)
+      value = value.replace(
+        /enemy:([a-zA-Z0-9_]+)[\.:]([a-zA-Z0-9_]+)/g,
+        "core.material.enemys['$1'].$2"
+      );
+    if (value.indexOf("blockId:") >= 0)
+      value = value.replace(/blockId:(\d+),(\d+)/g, "core.getBlockId($1, $2)");
+    if (value.indexOf("blockNumber:") >= 0)
+      value = value.replace(
+        /blockNumber:(\d+),(\d+)/g,
+        "core.getBlockNumber($1, $2)"
+      );
+    if (value.indexOf("blockCls:") >= 0)
+      value = value.replace(
+        /blockCls:(\d+),(\d+)/g,
+        "core.getBlockCls($1, $2)"
+      );
+    if (value.indexOf("equip:") >= 0)
+      value = value.replace(/equip:(\d)/g, "core.getEquip($1)");
+    if (value.indexOf("temp:") >= 0)
+      value = value.replace(
+        /temp:([a-zA-Z0-9_]+)/g,
+        "core.getFlag('@temp@$1', 0)"
+      );
+  }
+  return value;
+};
 
 ////// 计算表达式的值 //////
 utils.prototype.calValue = function (value, prefix) {
-    if (!core.isset(value)) return null;
-    if (typeof value === 'string') {
-        if (value.indexOf(':') >= 0 || value.indexOf("flag:") >= 0 || value.indexOf('global:') >= 0) {
-            if (value.indexOf('switch:') >= 0)
-                value = value.replace(/switch:([a-zA-Z0-9_]+)/g, "core.getFlag('" + (prefix || ":f@x@y") + "@$1', 0)");
-            value = this.replaceValue(value);
-        }
-        return eval(value);
+  if (!core.isset(value)) return null;
+  if (typeof value === "string") {
+    if (
+      value.indexOf(":") >= 0 ||
+      value.indexOf("flag:") >= 0 ||
+      value.indexOf("global:") >= 0
+    ) {
+      if (value.indexOf("switch:") >= 0)
+        value = value.replace(
+          /switch:([a-zA-Z0-9_]+)/g,
+          "core.getFlag('" + (prefix || ":f@x@y") + "@$1', 0)"
+        );
+      value = this.replaceValue(value);
     }
-    if (value instanceof Function) {
-        return value();
-    }
-    return value;
-}
+    return eval(value);
+  }
+  if (value instanceof Function) {
+    return value();
+  }
+  return value;
+};
 
 ////// 向某个数组前插入另一个数组或元素 //////
 utils.prototype.unshift = function (a, b) {
-    if (!(a instanceof Array) || b == null) return;
-    if (b instanceof Array) {
-        core.clone(b).reverse().forEach(function (e) {
-            a.unshift(e);
-        });
-    }
-    else a.unshift(b);
-    return a;
-}
+  if (!(a instanceof Array) || b == null) return;
+  if (b instanceof Array) {
+    core
+      .clone(b)
+      .reverse()
+      .forEach(function (e) {
+        a.unshift(e);
+      });
+  } else a.unshift(b);
+  return a;
+};
 
 ////// 向某个数组后插入另一个数组或元素 //////
 utils.prototype.push = function (a, b) {
-    if (!(a instanceof Array) || b == null) return;
-    if (b instanceof Array) {
-        core.clone(b).forEach(function (e) {
-            a.push(e);
-        });
-    }
-    else a.push(b);
-    return a;
-}
+  if (!(a instanceof Array) || b == null) return;
+  if (b instanceof Array) {
+    core.clone(b).forEach(function (e) {
+      a.push(e);
+    });
+  } else a.push(b);
+  return a;
+};
 
 utils.prototype.decompress = function (value) {
-    try {
-        var output = lzw_decode(value);
-        if (output) return JSON.parse(output);
-    }
-    catch (e) {
-    }
-    try {
-        var output = LZString.decompress(value);
-        if (output) return JSON.parse(output);
-    }
-    catch (e) {
-    }
-    try {
-        return JSON.parse(value);
-    }
-    catch (e) {
-        console.error(e);
-    }
-    return null;
-}
+  try {
+    var output = lzw_decode(value);
+    if (output) return JSON.parse(output);
+  } catch (e) {}
+  try {
+    var output = LZString.decompress(value);
+    if (output) return JSON.parse(output);
+  } catch (e) {}
+  try {
+    return JSON.parse(value);
+  } catch (e) {
+    console.error(e);
+  }
+  return null;
+};
 
 ////// 设置本地存储 //////
 utils.prototype.setLocalStorage = function (key, value) {
-    try {
-        if (value == null) {
-            this.removeLocalStorage(key);
-            return;
-        }
-
-        var str = JSON.stringify(value).replace(/[\u007F-\uFFFF]/g, function (chr) {
-            return "\\u" + ("0000" + chr.charCodeAt(0).toString(16)).substr(-4)
-        });
-        localStorage.setItem(core.firstData.name + "_" + key, str);
-
-        if (key == 'autoSave') core.saves.ids[0] = true;
-        else if (/^save\d+$/.test(key)) core.saves.ids[parseInt(key.substring(4))] = true;
-
-        return true;
+  try {
+    if (value == null) {
+      this.removeLocalStorage(key);
+      return;
     }
-    catch (e) {
-        console.error(e);
-        return false;
-    }
-}
+
+    var str = JSON.stringify(value).replace(/[\u007F-\uFFFF]/g, function (chr) {
+      return "\\u" + ("0000" + chr.charCodeAt(0).toString(16)).substr(-4);
+    });
+    localStorage.setItem(core.firstData.name + "_" + key, str);
+
+    if (key == "autoSave") core.saves.ids[0] = true;
+    else if (/^save\d+$/.test(key))
+      core.saves.ids[parseInt(key.substring(4))] = true;
+
+    return true;
+  } catch (e) {
+    console.error(e);
+    return false;
+  }
+};
 
 ////// 获得本地存储 //////
 utils.prototype.getLocalStorage = function (key, defaultValue) {
-    try {
-        var value = JSON.parse(localStorage.getItem(core.firstData.name + "_" + key));
-        if (value == null) return defaultValue;
-        return value;
-    } catch (e) {
-        return defaultValue;
-    }
-}
+  try {
+    var value = JSON.parse(
+      localStorage.getItem(core.firstData.name + "_" + key)
+    );
+    if (value == null) return defaultValue;
+    return value;
+  } catch (e) {
+    return defaultValue;
+  }
+};
 
 ////// 移除本地存储 //////
 utils.prototype.removeLocalStorage = function (key) {
-    localStorage.removeItem(core.firstData.name + "_" + key);
-    if (key == 'autoSave') delete core.saves.ids[0];
-    else if (/^save\d+$/.test(key)) delete core.saves.ids[parseInt(key.substring(4))];
-}
+  localStorage.removeItem(core.firstData.name + "_" + key);
+  if (key == "autoSave") delete core.saves.ids[0];
+  else if (/^save\d+$/.test(key))
+    delete core.saves.ids[parseInt(key.substring(4))];
+};
 
-utils.prototype.setLocalForage = function (key, value, successCallback, errorCallback) {
-    if (value == null) {
-        this.removeLocalForage(key);
-        return;
-    }
+utils.prototype.setLocalForage = function (
+  key,
+  value,
+  successCallback,
+  errorCallback
+) {
+  if (value == null) {
+    this.removeLocalForage(key);
+    return;
+  }
 
-    var name = core.firstData.name + "_" + key;
-    var str = JSON.stringify(value).replace(/[\u007F-\uFFFF]/g, function (chr) {
-        return "\\u" + ("0000" + chr.charCodeAt(0).toString(16)).substr(-4)
-    });
-    var callback = function (err) {
-        if (err) {
-            if (errorCallback) errorCallback(err);
-        }
-        else {
-            if (key == 'autoSave') core.saves.ids[0] = true;
-            else if (/^save\d+$/.test(key)) core.saves.ids[parseInt(key.substring(4))] = true;
-            if (successCallback) successCallback();
-        }
+  var name = core.firstData.name + "_" + key;
+  var str = JSON.stringify(value).replace(/[\u007F-\uFFFF]/g, function (chr) {
+    return "\\u" + ("0000" + chr.charCodeAt(0).toString(16)).substr(-4);
+  });
+  var callback = function (err) {
+    if (err) {
+      if (errorCallback) errorCallback(err);
+    } else {
+      if (key == "autoSave") core.saves.ids[0] = true;
+      else if (/^save\d+$/.test(key))
+        core.saves.ids[parseInt(key.substring(4))] = true;
+      if (successCallback) successCallback();
     }
-    this._setLocalForage_set(name, str, callback);
-}
+  };
+  this._setLocalForage_set(name, str, callback);
+};
 
 utils.prototype._setLocalForage_set = function (name, str, callback) {
-    if (window.jsinterface && window.jsinterface.setLocalForage) {
-        var id = setTimeout(null);
-        core['__callback' + id] = callback;
-        core.saves.cache[name] = str;
-        window.jsinterface.setLocalForage(id, name, str);
-    } else {
-        var compressed = str.length > 100000 ? LZString.compress(str) : lzw_encode(str);
-        core.saves.cache[name] = compressed;
-        localforage.setItem(name, compressed, callback);
-    }
-}
+  if (window.jsinterface && window.jsinterface.setLocalForage) {
+    var id = setTimeout(null);
+    core["__callback" + id] = callback;
+    core.saves.cache[name] = str;
+    window.jsinterface.setLocalForage(id, name, str);
+  } else {
+    var compressed =
+      str.length > 100000 ? LZString.compress(str) : lzw_encode(str);
+    core.saves.cache[name] = compressed;
+    localforage.setItem(name, compressed, callback);
+  }
+};
 
-utils.prototype.getLocalForage = function (key, defaultValue, successCallback, errorCallback) {
-    var name = core.firstData.name + "_" + key;
-    var callback = function (err, value) {
-        if (err) {
-            if (errorCallback) errorCallback(err);
-        }
-        else {
-            core.saves.cache[name] = value;
-            if (!successCallback) return;
-            if (value != null) {
-                var res = core.utils.decompress(value);
-                successCallback(res == null ? defaultValue : res);
-                return;
-            }
-            successCallback(defaultValue);
-        }
-    };
-    if (core.saves.cache[name] != null) {
-        return callback(null, core.saves.cache[name]);
+utils.prototype.getLocalForage = function (
+  key,
+  defaultValue,
+  successCallback,
+  errorCallback
+) {
+  var name = core.firstData.name + "_" + key;
+  var callback = function (err, value) {
+    if (err) {
+      if (errorCallback) errorCallback(err);
+    } else {
+      core.saves.cache[name] = value;
+      if (!successCallback) return;
+      if (value != null) {
+        var res = core.utils.decompress(value);
+        successCallback(res == null ? defaultValue : res);
+        return;
+      }
+      successCallback(defaultValue);
     }
-    this._getLocalForage_get(name, callback);
-}
+  };
+  if (core.saves.cache[name] != null) {
+    return callback(null, core.saves.cache[name]);
+  }
+  this._getLocalForage_get(name, callback);
+};
 
 utils.prototype._getLocalForage_get = function (name, callback) {
-    if (window.jsinterface && window.jsinterface.getLocalForage) {
-        var id = setTimeout(null);
-        core['__callback' + id] = callback;
-        window.jsinterface.getLocalForage(id, name);
-    } else {
-        localforage.getItem(name, callback);
-    }
-}
+  if (window.jsinterface && window.jsinterface.getLocalForage) {
+    var id = setTimeout(null);
+    core["__callback" + id] = callback;
+    window.jsinterface.getLocalForage(id, name);
+  } else {
+    localforage.getItem(name, callback);
+  }
+};
 
-utils.prototype.removeLocalForage = function (key, successCallback, errorCallback) {
-    var name = core.firstData.name + "_" + key;
-    var callback = function (err) {
-        if (err) {
-            if (errorCallback) errorCallback(err);
-        }
-        else {
-            if (key == 'autoSave') delete core.saves.ids[0];
-            else if (/^save\d+$/.test(key)) delete core.saves.ids[parseInt(key.substring(4))];
-            if (successCallback) successCallback();
-        }
+utils.prototype.removeLocalForage = function (
+  key,
+  successCallback,
+  errorCallback
+) {
+  var name = core.firstData.name + "_" + key;
+  var callback = function (err) {
+    if (err) {
+      if (errorCallback) errorCallback(err);
+    } else {
+      if (key == "autoSave") delete core.saves.ids[0];
+      else if (/^save\d+$/.test(key))
+        delete core.saves.ids[parseInt(key.substring(4))];
+      if (successCallback) successCallback();
     }
-    delete core.saves.cache[name];
-    this._removeLocalForage_remove(name, callback);
-}
+  };
+  delete core.saves.cache[name];
+  this._removeLocalForage_remove(name, callback);
+};
 
 utils.prototype._removeLocalForage_remove = function (name, callback) {
-    if (window.jsinterface && window.jsinterface.removeLocalForage) {
-        var id = setTimeout(null);
-        core['__callback' + id] = callback;
-        window.jsinterface.removeLocalForage(id, name);
-    } else {
-        localforage.removeItem(name, callback);
-    }
-}
+  if (window.jsinterface && window.jsinterface.removeLocalForage) {
+    var id = setTimeout(null);
+    core["__callback" + id] = callback;
+    window.jsinterface.removeLocalForage(id, name);
+  } else {
+    localforage.removeItem(name, callback);
+  }
+};
 
 utils.prototype.clearLocalForage = function (callback) {
-    core.saves.cache = {};
-    if (window.jsinterface && window.jsinterface.clearLocalForage) {
-        var id = setTimeout(null);
-        core['__callback' + id] = callback;
-        window.jsinterface.clearLocalForage(id);
-    } else {
-        localforage.clear(callback);
-    }
-}
+  core.saves.cache = {};
+  if (window.jsinterface && window.jsinterface.clearLocalForage) {
+    var id = setTimeout(null);
+    core["__callback" + id] = callback;
+    window.jsinterface.clearLocalForage(id);
+  } else {
+    localforage.clear(callback);
+  }
+};
 
 utils.prototype.iterateLocalForage = function (iter, callback) {
-    if (window.jsinterface && window.jsinterface.iterateLocalForage) {
-        var id = setTimeout(null);
-        core['__iter' + id] = iter;
-        core['__callback' + id] = callback;
-        window.jsinterface.iterateLocalForage(id);
-    } else {
-        localforage.iterate(iter, callback);
-    }
-}
+  if (window.jsinterface && window.jsinterface.iterateLocalForage) {
+    var id = setTimeout(null);
+    core["__iter" + id] = iter;
+    core["__callback" + id] = callback;
+    window.jsinterface.iterateLocalForage(id);
+  } else {
+    localforage.iterate(iter, callback);
+  }
+};
 
 utils.prototype.keysLocalForage = function (callback) {
-    if (window.jsinterface && window.jsinterface.keysLocalForage) {
-        var id = setTimeout(null);
-        core['__callback' + id] = callback;
-        window.jsinterface.keysLocalForage(id);
-    } else {
-        localforage.keys(callback);
-    }
-}
+  if (window.jsinterface && window.jsinterface.keysLocalForage) {
+    var id = setTimeout(null);
+    core["__callback" + id] = callback;
+    window.jsinterface.keysLocalForage(id);
+  } else {
+    localforage.keys(callback);
+  }
+};
 
 utils.prototype.lengthLocalForage = function (callback) {
-    if (window.jsinterface && window.jsinterface.lengthLocalForage) {
-        var id = setTimeout(null);
-        core['__callback' + id] = callback;
-        window.jsinterface.lengthLocalForage(id);
-    } else {
-        localforage.length(callback);
-    }
-}
+  if (window.jsinterface && window.jsinterface.lengthLocalForage) {
+    var id = setTimeout(null);
+    core["__callback" + id] = callback;
+    window.jsinterface.lengthLocalForage(id);
+  } else {
+    localforage.length(callback);
+  }
+};
 
 utils.prototype.setGlobal = function (key, value) {
-    if (core.isReplaying()) return;
-    core.setLocalStorage(key, value);
-}
+  if (core.isReplaying()) return;
+  core.setLocalStorage(key, value);
+};
 
 utils.prototype.getGlobal = function (key, defaultValue) {
-    var value;
-    if (core.isReplaying()) {
-        // 不考虑key不一致的情况
-        var action = core.status.replay.toReplay.shift();
-        if (action.indexOf("input2:") == 0) {
-            value = JSON.parse(core.decodeBase64(action.substring(7)));
-            core.setFlag('__global__' + key, value);
-            core.status.route.push("input2:" + core.encodeBase64(JSON.stringify(value)));
-        }
-        else {
-            // 录像兼容性:尝试从flag和localStorage获得
-            // 注意这里不再二次记录 input2: 到录像
-            core.status.replay.toReplay.unshift(action);
-            value = core.getFlag('__global__' + key, core.getLocalStorage(key, defaultValue));
-        }
+  var value;
+  if (core.isReplaying()) {
+    // 不考虑key不一致的情况
+    var action = core.status.replay.toReplay.shift();
+    if (action.indexOf("input2:") == 0) {
+      value = JSON.parse(core.decodeBase64(action.substring(7)));
+      core.setFlag("__global__" + key, value);
+      core.status.route.push(
+        "input2:" + core.encodeBase64(JSON.stringify(value))
+      );
+    } else {
+      // 录像兼容性:尝试从flag和localStorage获得
+      // 注意这里不再二次记录 input2: 到录像
+      core.status.replay.toReplay.unshift(action);
+      value = core.getFlag(
+        "__global__" + key,
+        core.getLocalStorage(key, defaultValue)
+      );
     }
-    else {
-        value = core.getLocalStorage(key, defaultValue);
-        core.setFlag('__global__' + key, value);
-        core.status.route.push("input2:" + core.encodeBase64(JSON.stringify(value)));
-    }
-    return value;
-}
+  } else {
+    value = core.getLocalStorage(key, defaultValue);
+    core.setFlag("__global__" + key, value);
+    core.status.route.push(
+      "input2:" + core.encodeBase64(JSON.stringify(value))
+    );
+  }
+  return value;
+};
 
 ////// 深拷贝一个对象 //////
 utils.prototype.clone = function (data, filter, recursion) {
-    if (!core.isset(data)) return null;
-    // date
-    if (data instanceof Date) {
-        var copy = new Date();
-        copy.setTime(data.getTime());
-        return copy;
-    }
-    // array
-    if (data instanceof Array) {
-        var copy = [];
-        for (var i in data) {
-            if (!filter || filter(i, data[i]))
-                copy[i] = core.clone(data[i], recursion ? filter : null, recursion);
-        }
-        return copy;
-    }
-    // 函数
-    if (data instanceof Function) {
-        return data;
-    }
-    // object
-    if (data instanceof Object) {
-        var copy = {};
-        for (var i in data) {
-            if (data.hasOwnProperty(i) && (!filter || filter(i, data[i])))
-                copy[i] = core.clone(data[i], recursion ? filter : null, recursion);
-        }
-        return copy;
+  if (!core.isset(data)) return null;
+  // date
+  if (data instanceof Date) {
+    var copy = new Date();
+    copy.setTime(data.getTime());
+    return copy;
+  }
+  // array
+  if (data instanceof Array) {
+    var copy = [];
+    for (var i in data) {
+      if (!filter || filter(i, data[i]))
+        copy[i] = core.clone(data[i], recursion ? filter : null, recursion);
     }
+    return copy;
+  }
+  // 函数
+  if (data instanceof Function) {
     return data;
-}
+  }
+  // object
+  if (data instanceof Object) {
+    var copy = {};
+    for (var i in data) {
+      if (data.hasOwnProperty(i) && (!filter || filter(i, data[i])))
+        copy[i] = core.clone(data[i], recursion ? filter : null, recursion);
+    }
+    return copy;
+  }
+  return data;
+};
 
 ////// 深拷贝1D/2D数组优化 //////
 utils.prototype.cloneArray = function (data) {
-    if (!(data instanceof Array)) return this.clone(data);
-    if (data[0] instanceof Array) {
-        return data.map(function (one) { return one.slice(); });
-    } else {
-        return data.slice();
-    }
-}
+  if (!(data instanceof Array)) return this.clone(data);
+  if (data[0] instanceof Array) {
+    return data.map(function (one) {
+      return one.slice();
+    });
+  } else {
+    return data.slice();
+  }
+};
 
 ////// 裁剪图片 //////
 utils.prototype.splitImage = function (image, width, height) {
-    if (typeof image == "string") {
-        image = core.getMappedName(image);
-        image = core.material.images.images[image];
+  if (typeof image == "string") {
+    image = core.getMappedName(image);
+    image = core.material.images.images[image];
+  }
+  if (!image) return [];
+  width = width || 32;
+  height = height || width;
+  var canvas = document.createElement("canvas");
+  var ctx = canvas.getContext("2d");
+  var ans = [];
+  for (var j = 0; j < image.height; j += height) {
+    for (var i = 0; i < image.width; i += width) {
+      var w = Math.min(width, image.width - i),
+        h = Math.min(height, image.height - j);
+      canvas.width = w;
+      canvas.height = h;
+      core.drawImage(ctx, image, i, j, w, h, 0, 0, w, h);
+      var img = new Image();
+      img.src = canvas.toDataURL("image/png");
+      ans.push(img);
     }
-    if (!image) return [];
-    width = width || 32;
-    height = height || width;
-    var canvas = document.createElement("canvas");
-    var ctx = canvas.getContext("2d");
-    var ans = [];
-    for (var j = 0; j < image.height; j += height) {
-        for (var i = 0; i < image.width; i += width) {
-            var w = Math.min(width, image.width - i), h = Math.min(height, image.height - j);
-            canvas.width = w; canvas.height = h;
-            core.drawImage(ctx, image, i, j, w, h, 0, 0, w, h);
-            var img = new Image();
-            img.src = canvas.toDataURL("image/png");
-            ans.push(img);
-        }
-    }
-    return ans;
-}
+  }
+  return ans;
+};
 
 ////// 格式化时间为字符串 //////
 utils.prototype.formatDate = function (date) {
-    if (!date) date = new Date();
-    return "" + date.getFullYear() + "-" + core.setTwoDigits(date.getMonth() + 1) + "-" + core.setTwoDigits(date.getDate()) + " "
-        + core.setTwoDigits(date.getHours()) + ":" + core.setTwoDigits(date.getMinutes()) + ":" + core.setTwoDigits(date.getSeconds());
-}
+  if (!date) date = new Date();
+  return (
+    "" +
+    date.getFullYear() +
+    "-" +
+    core.setTwoDigits(date.getMonth() + 1) +
+    "-" +
+    core.setTwoDigits(date.getDate()) +
+    " " +
+    core.setTwoDigits(date.getHours()) +
+    ":" +
+    core.setTwoDigits(date.getMinutes()) +
+    ":" +
+    core.setTwoDigits(date.getSeconds())
+  );
+};
 
 ////// 格式化时间为最简字符串 //////
 utils.prototype.formatDate2 = function (date) {
-    if (!date) date = new Date();
-    return "" + date.getFullYear() + core.setTwoDigits(date.getMonth() + 1) + core.setTwoDigits(date.getDate())
-        + core.setTwoDigits(date.getHours()) + core.setTwoDigits(date.getMinutes()) + core.setTwoDigits(date.getSeconds());
-}
+  if (!date) date = new Date();
+  return (
+    "" +
+    date.getFullYear() +
+    core.setTwoDigits(date.getMonth() + 1) +
+    core.setTwoDigits(date.getDate()) +
+    core.setTwoDigits(date.getHours()) +
+    core.setTwoDigits(date.getMinutes()) +
+    core.setTwoDigits(date.getSeconds())
+  );
+};
 
 utils.prototype.formatTime = function (time) {
-    return core.setTwoDigits(parseInt(time / 3600000))
-        + ":" + core.setTwoDigits(parseInt(time / 60000) % 60)
-        + ":" + core.setTwoDigits(parseInt(time / 1000) % 60);
-}
+  return (
+    core.setTwoDigits(parseInt(time / 3600000)) +
+    ":" +
+    core.setTwoDigits(parseInt(time / 60000) % 60) +
+    ":" +
+    core.setTwoDigits(parseInt(time / 1000) % 60)
+  );
+};
 
 ////// 两位数显示 //////
 utils.prototype.setTwoDigits = function (x) {
-    return (parseInt(x) < 10 && parseInt(x) >= 0) ? "0" + x : x;
-}
+  return parseInt(x) < 10 && parseInt(x) >= 0 ? "0" + x : x;
+};
 
 utils.prototype.formatSize = function (size) {
-    if (size < 1024) return size + 'B';
-    else if (size < 1024 * 1024) return (size / 1024).toFixed(2) + "KB";
-    else return (size / 1024 / 1024).toFixed(2) + "MB";
-}
+  if (size < 1024) return size + "B";
+  else if (size < 1024 * 1024) return (size / 1024).toFixed(2) + "KB";
+  else return (size / 1024 / 1024).toFixed(2) + "MB";
+};
 
 utils.prototype.formatBigNumber = function (x, digits) {
-    if (digits === true) digits = 5; // 兼容旧版onMap参数
-    if (!digits || digits < 5) digits = 6; // 连同负号、小数点和后缀字母在内的总位数,至少需为5,默认为6
-    x = Math.trunc(parseFloat(x)); // 尝试识别为小数,然后向0取整
-    if (x == null || !Number.isFinite(x)) return '???'; // 无法识别的数或正负无穷大,显示'???'
-    var units = [ // 单位及其后缀字母,可自定义,如改成千进制下的K、M、G、T、P
-        { "val": 1e4, "suffix": "w" },
-        { "val": 1e8, "suffix": "e" },
-        { "val": 1e12, "suffix": "z" },
-        { "val": 1e16, "suffix": "j" },
-        { "val": 1e20, "suffix": "g" },
-    ];
-    if (Math.abs(x) > 1e20 * Math.pow(10, digits - 2))
-        return x.toExponential(0); // 绝对值过大以致于失去精度的数,直接使用科学记数法,系数只保留整数
-    var sign = x < 0 ? '-' : '';
-    if (sign) --digits; // 符号位单独处理,负号要占一位
-    x = Math.abs(x);
+  if (digits === true) digits = 5; // 兼容旧版onMap参数
+  if (!digits || digits < 5) digits = 6; // 连同负号、小数点和后缀字母在内的总位数,至少需为5,默认为6
+  x = Math.trunc(parseFloat(x)); // 尝试识别为小数,然后向0取整
+  if (x == null || !Number.isFinite(x)) return "???"; // 无法识别的数或正负无穷大,显示'???'
+  var units = [
+    // 单位及其后缀字母,可自定义,如改成千进制下的K、M、G、T、P
+    { val: 1e4, suffix: "w" },
+    { val: 1e8, suffix: "e" },
+    { val: 1e12, suffix: "z" },
+    { val: 1e16, suffix: "j" },
+    { val: 1e20, suffix: "g" },
+  ];
+  if (Math.abs(x) > 1e20 * Math.pow(10, digits - 2)) return x.toExponential(0); // 绝对值过大以致于失去精度的数,直接使用科学记数法,系数只保留整数
+  var sign = x < 0 ? "-" : "";
+  if (sign) --digits; // 符号位单独处理,负号要占一位
+  x = Math.abs(x);
 
-    if (x < Math.pow(10, digits)) return sign + x;
+  if (x < Math.pow(10, digits)) return sign + x;
 
-    for (var i = 0; i < units.length; ++i) {
-        var each = units[i];
-        var u = (x / each.val).toFixed(digits).substring(0, digits);
-        if (u.indexOf('.') < 0) continue;
-        u = u.substring(0, u[u.length - 2] == '.' ? u.length - 2 : u.length - 1);
-        return sign + u + each.suffix;
-    }
-    return sign + x.toExponential(0);
-}
+  for (var i = 0; i < units.length; ++i) {
+    var each = units[i];
+    var u = (x / each.val).toFixed(digits).substring(0, digits);
+    if (u.indexOf(".") < 0) continue;
+    u = u.substring(0, u[u.length - 2] == "." ? u.length - 2 : u.length - 1);
+    return sign + u + each.suffix;
+  }
+  return sign + x.toExponential(0);
+};
 
 ////// 变速移动 //////
 utils.prototype.applyEasing = function (name) {
-    var list = {
-        "easeIn": function (t) {
-            return Math.pow(t, 3);
-        },
-        "easeOut": function (t) {
-            return 1 - Math.pow(1 - t, 3);
-        },
-        "easeInOut": function (t) {
-            // easeInOut试了一下感觉二次方效果明显点
-            if (t < 0.5) return Math.pow(t, 2) * 2;
-            else return 1 - Math.pow(1 - t, 2) * 2;
-        },
-        "linear": function (t) {
-            return t
-        }
-    }
-    if (name == 'random') {
-        var keys = Object.keys(list);
-        name = keys[Math.floor(Math.random() * keys.length)];
-    }
-    return list[name] || list.linear;
-}
+  var list = {
+    easeIn: function (t) {
+      return Math.pow(t, 3);
+    },
+    easeOut: function (t) {
+      return 1 - Math.pow(1 - t, 3);
+    },
+    easeInOut: function (t) {
+      // easeInOut试了一下感觉二次方效果明显点
+      if (t < 0.5) return Math.pow(t, 2) * 2;
+      else return 1 - Math.pow(1 - t, 2) * 2;
+    },
+    linear: function (t) {
+      return t;
+    },
+  };
+  if (name == "random") {
+    var keys = Object.keys(list);
+    name = keys[Math.floor(Math.random() * keys.length)];
+  }
+  return list[name] || list.linear;
+};
 
 ////// 数组转RGB //////
 utils.prototype.arrayToRGB = function (color) {
-    if (!(color instanceof Array)) return color;
-    var nowR = this.clamp(parseInt(color[0]), 0, 255), nowG = this.clamp(parseInt(color[1]), 0, 255),
-        nowB = this.clamp(parseInt(color[2]), 0, 255);
-    return "#" + ((1 << 24) + (nowR << 16) + (nowG << 8) + nowB).toString(16).slice(1);
-}
+  if (!(color instanceof Array)) return color;
+  var nowR = this.clamp(parseInt(color[0]), 0, 255),
+    nowG = this.clamp(parseInt(color[1]), 0, 255),
+    nowB = this.clamp(parseInt(color[2]), 0, 255);
+  return (
+    "#" + ((1 << 24) + (nowR << 16) + (nowG << 8) + nowB).toString(16).slice(1)
+  );
+};
 
 utils.prototype.arrayToRGBA = function (color) {
-    if (!(color instanceof Array)) return color;
-    if (color[3] == null) color[3] = 1;
-    var nowR = this.clamp(parseInt(color[0]), 0, 255), nowG = this.clamp(parseInt(color[1]), 0, 255),
-        nowB = this.clamp(parseInt(color[2]), 0, 255), nowA = this.clamp(parseFloat(color[3]), 0, 1);
-    return "rgba(" + nowR + "," + nowG + "," + nowB + "," + nowA + ")";
-}
+  if (!(color instanceof Array)) return color;
+  if (color[3] == null) color[3] = 1;
+  var nowR = this.clamp(parseInt(color[0]), 0, 255),
+    nowG = this.clamp(parseInt(color[1]), 0, 255),
+    nowB = this.clamp(parseInt(color[2]), 0, 255),
+    nowA = this.clamp(parseFloat(color[3]), 0, 1);
+  return "rgba(" + nowR + "," + nowG + "," + nowB + "," + nowA + ")";
+};
 
 ////// 加密路线 //////
 utils.prototype.encodeRoute = function (route) {
-    var ans = "", lastMove = "", cnt = 0;
+  var ans = "",
+    lastMove = "",
+    cnt = 0;
 
-    route.forEach(function (t) {
-        if (t == 'up' || t == 'down' || t == 'left' || t == 'right') {
-            if (t != lastMove && cnt > 0) {
-                ans += lastMove.substring(0, 1).toUpperCase();
-                if (cnt > 1) ans += cnt;
-                cnt = 0;
-            }
-            lastMove = t;
-            cnt++;
-        }
-        else {
-            if (cnt > 0) {
-                ans += lastMove.substring(0, 1).toUpperCase();
-                if (cnt > 1) ans += cnt;
-                cnt = 0;
-            }
-            ans += core.utils._encodeRoute_encodeOne(t);
-        }
-    });
-    if (cnt > 0) {
+  route.forEach(function (t) {
+    if (t == "up" || t == "down" || t == "left" || t == "right") {
+      if (t != lastMove && cnt > 0) {
         ans += lastMove.substring(0, 1).toUpperCase();
         if (cnt > 1) ans += cnt;
+        cnt = 0;
+      }
+      lastMove = t;
+      cnt++;
+    } else {
+      if (cnt > 0) {
+        ans += lastMove.substring(0, 1).toUpperCase();
+        if (cnt > 1) ans += cnt;
+        cnt = 0;
+      }
+      ans += core.utils._encodeRoute_encodeOne(t);
     }
-    return LZString.compressToBase64(ans);
-}
+  });
+  if (cnt > 0) {
+    ans += lastMove.substring(0, 1).toUpperCase();
+    if (cnt > 1) ans += cnt;
+  }
+  return LZString.compressToBase64(ans);
+};
 
 utils.prototype._encodeRoute_id2number = function (id) {
-    var number = core.maps.getNumberById(id);
-    return number == 0 ? id : number;
-}
+  var number = core.maps.getNumberById(id);
+  return number == 0 ? id : number;
+};
 
 utils.prototype._encodeRoute_encodeOne = function (t) {
-    if (t.indexOf('item:') == 0)
-        return "I" + this._encodeRoute_id2number(t.substring(5)) + ":";
-    else if (t.indexOf('unEquip:') == 0)
-        return "u" + t.substring(8);
-    else if (t.indexOf('equip:') == 0)
-        return "e" + this._encodeRoute_id2number(t.substring(6)) + ":";
-    else if (t.indexOf('saveEquip:') == 0)
-        return "s" + t.substring(10);
-    else if (t.indexOf('loadEquip:') == 0)
-        return "l" + t.substring(10);
-    else if (t.indexOf('fly:') == 0)
-        return "F" + t.substring(4) + ":";
-    else if (t == 'choices:none')
-        return "c";
-    else if (t.indexOf('choices:') == 0)
-        return "C" + t.substring(8);
-    else if (t.indexOf('shop:') == 0)
-        return "S" + t.substring(5) + ":";
-    else if (t == 'turn')
-        return 'T';
-    else if (t.indexOf('turn:') == 0)
-        return "t" + t.substring(5).substring(0, 1).toUpperCase() + ":";
-    else if (t == 'getNext')
-        return 'G';
-    else if (t == 'input:none')
-        return 'p';
-    else if (t.indexOf('input:') == 0)
-        return "P" + t.substring(6);
-    else if (t.indexOf('input2:') == 0)
-        return "Q" + t.substring(7) + ":";
-    else if (t == 'no')
-        return 'N';
-    else if (t.indexOf('move:') == 0)
-        return "M" + t.substring(5);
-    else if (t.indexOf('key:') == 0)
-        return 'K' + t.substring(4);
-    else if (t.indexOf('click:') == 0)
-        return 'k' + t.substring(6);
-    else if (t.indexOf('random:') == 0)
-        return 'X' + t.substring(7);
-    return '(' + t + ')';
-}
+  if (t.indexOf("item:") == 0)
+    return "I" + this._encodeRoute_id2number(t.substring(5)) + ":";
+  else if (t.indexOf("unEquip:") == 0) return "u" + t.substring(8);
+  else if (t.indexOf("equip:") == 0)
+    return "e" + this._encodeRoute_id2number(t.substring(6)) + ":";
+  else if (t.indexOf("saveEquip:") == 0) return "s" + t.substring(10);
+  else if (t.indexOf("loadEquip:") == 0) return "l" + t.substring(10);
+  else if (t.indexOf("fly:") == 0) return "F" + t.substring(4) + ":";
+  else if (t == "choices:none") return "c";
+  else if (t.indexOf("choices:") == 0) return "C" + t.substring(8);
+  else if (t.indexOf("shop:") == 0) return "S" + t.substring(5) + ":";
+  else if (t == "turn") return "T";
+  else if (t.indexOf("turn:") == 0)
+    return "t" + t.substring(5).substring(0, 1).toUpperCase() + ":";
+  else if (t == "getNext") return "G";
+  else if (t == "input:none") return "p";
+  else if (t.indexOf("input:") == 0) return "P" + t.substring(6);
+  else if (t.indexOf("input2:") == 0) return "Q" + t.substring(7) + ":";
+  else if (t == "no") return "N";
+  else if (t.indexOf("move:") == 0) return "M" + t.substring(5);
+  else if (t.indexOf("key:") == 0) return "K" + t.substring(4);
+  else if (t.indexOf("click:") == 0) return "k" + t.substring(6);
+  else if (t.indexOf("random:") == 0) return "X" + t.substring(7);
+  return "(" + t + ")";
+};
 
 ////// 解密路线 //////
 utils.prototype.decodeRoute = function (route) {
-    if (!route) return route;
+  if (!route) return route;
 
-    // 解压缩
-    try {
-        var v = LZString.decompressFromBase64(route);
-        if (v != null && /^[-_a-zA-Z0-9+\/=:()]*$/.test(v)) {
-            if (v != "" || route.length < 8)
-                route = v;
-        }
-    } catch (e) {
+  // 解压缩
+  try {
+    var v = LZString.decompressFromBase64(route);
+    if (v != null && /^[-_a-zA-Z0-9+\/=:()]*$/.test(v)) {
+      if (v != "" || route.length < 8) route = v;
     }
+  } catch (e) {}
 
-    var decodeObj = { route: route, index: 0, ans: [] };
-    while (decodeObj.index < decodeObj.route.length) {
-        this._decodeRoute_decodeOne(decodeObj, decodeObj.route.charAt(decodeObj.index++));
-    }
-    return decodeObj.ans;
-}
+  var decodeObj = { route: route, index: 0, ans: [] };
+  while (decodeObj.index < decodeObj.route.length) {
+    this._decodeRoute_decodeOne(
+      decodeObj,
+      decodeObj.route.charAt(decodeObj.index++)
+    );
+  }
+  return decodeObj.ans;
+};
 
 utils.prototype._decodeRoute_getNumber = function (decodeObj, noparse) {
-    var num = "";
-    var first = true;
-    while (true) {
-        var ch = decodeObj.route.charAt(decodeObj.index);
-        if (ch >= '0' && ch <= '9') num += ch;
-        else if (ch == '-' && first) num += ch;
-        else break;
-        first = false;
-        decodeObj.index++;
-    }
-    if (num.length == 0) num = "1";
-    return noparse ? num : parseInt(num);
-}
+  var num = "";
+  var first = true;
+  while (true) {
+    var ch = decodeObj.route.charAt(decodeObj.index);
+    if (ch >= "0" && ch <= "9") num += ch;
+    else if (ch == "-" && first) num += ch;
+    else break;
+    first = false;
+    decodeObj.index++;
+  }
+  if (num.length == 0) num = "1";
+  return noparse ? num : parseInt(num);
+};
 
 utils.prototype._decodeRoute_getString = function (decodeObj) {
-    var str = "";
-    while (decodeObj.index < decodeObj.route.length && decodeObj.route.charAt(decodeObj.index) != ':') {
-        str += decodeObj.route.charAt(decodeObj.index++);
-    }
-    decodeObj.index++;
-    return str;
-}
+  var str = "";
+  while (
+    decodeObj.index < decodeObj.route.length &&
+    decodeObj.route.charAt(decodeObj.index) != ":"
+  ) {
+    str += decodeObj.route.charAt(decodeObj.index++);
+  }
+  decodeObj.index++;
+  return str;
+};
 
 utils.prototype._decodeRoute_number2id = function (number) {
-    if (/^\d+$/.test(number)) {
-        var info = core.maps.blocksInfo[number];
-        if (info) return info.id;
-    }
-    return number;
-}
+  if (/^\d+$/.test(number)) {
+    var info = core.maps.blocksInfo[number];
+    if (info) return info.id;
+  }
+  return number;
+};
 
 utils.prototype._decodeRoute_decodeOne = function (decodeObj, c) {
-    // --- 特殊处理自定义项
-    if (c == '(') {
-        var idx = decodeObj.route.indexOf(')', decodeObj.index);
-        if (idx >= 0) {
-            decodeObj.ans.push(decodeObj.route.substring(decodeObj.index, idx));
-            decodeObj.index = idx + 1;
-            return;
-        }
+  // --- 特殊处理自定义项
+  if (c == "(") {
+    var idx = decodeObj.route.indexOf(")", decodeObj.index);
+    if (idx >= 0) {
+      decodeObj.ans.push(decodeObj.route.substring(decodeObj.index, idx));
+      decodeObj.index = idx + 1;
+      return;
     }
-    var nxt = (c == 'I' || c == 'e' || c == 'F' || c == 'S' || c == 'Q' || c == 't') ?
-        this._decodeRoute_getString(decodeObj) : this._decodeRoute_getNumber(decodeObj);
+  }
+  var nxt =
+    c == "I" || c == "e" || c == "F" || c == "S" || c == "Q" || c == "t"
+      ? this._decodeRoute_getString(decodeObj)
+      : this._decodeRoute_getNumber(decodeObj);
 
-    var mp = { "U": "up", "D": "down", "L": "left", "R": "right" };
+  var mp = { U: "up", D: "down", L: "left", R: "right" };
 
-    switch (c) {
-        case "U":
-        case "D":
-        case "L":
-        case "R":
-            for (var i = 0; i < nxt; i++) decodeObj.ans.push(mp[c]);
-            break;
-        case "I":
-            decodeObj.ans.push("item:" + this._decodeRoute_number2id(nxt));
-            break;
-        case "u":
-            decodeObj.ans.push("unEquip:" + nxt);
-            break;
-        case "e":
-            decodeObj.ans.push("equip:" + this._decodeRoute_number2id(nxt));
-            break;
-        case "s":
-            decodeObj.ans.push("saveEquip:" + nxt);
-            break;
-        case "l":
-            decodeObj.ans.push("loadEquip:" + nxt);
-            break;
-        case "F":
-            decodeObj.ans.push("fly:" + nxt);
-            break;
-        case 'c':
-            decodeObj.ans.push('choices:none');
-            break;
-        case "C":
-            decodeObj.ans.push("choices:" + nxt);
-            break;
-        case "S":
-            decodeObj.ans.push("shop:" + nxt);
-            break;
-        case "T":
-            decodeObj.ans.push("turn");
-            break;
-        case "t":
-            decodeObj.ans.push("turn:" + mp[nxt]);
-            break;
-        case "G":
-            decodeObj.ans.push("getNext");
-            break;
-        case "p":
-            decodeObj.ans.push("input:none");
-            break;
-        case "P":
-            decodeObj.ans.push("input:" + nxt);
-            break;
-        case "Q":
-            decodeObj.ans.push("input2:" + nxt);
-            break;
-        case "N":
-            decodeObj.ans.push("no");
-            break;
-        case "M":
-            ++decodeObj.index;
-            decodeObj.ans.push("move:" + nxt + ":" + this._decodeRoute_getNumber(decodeObj));
-            break;
-        case "K":
-            decodeObj.ans.push("key:" + nxt);
-            break;
-        case "k":
-            ++decodeObj.index;
-            var px = this._decodeRoute_getNumber(decodeObj);
-            ++decodeObj.index;
-            var py = this._decodeRoute_getNumber(decodeObj);
-            decodeObj.ans.push("click:" + nxt + ":" + px + ":" + py);
-            break;
-        case "X":
-            decodeObj.ans.push("random:" + nxt);
-            break;
-    }
-}
+  switch (c) {
+    case "U":
+    case "D":
+    case "L":
+    case "R":
+      for (var i = 0; i < nxt; i++) decodeObj.ans.push(mp[c]);
+      break;
+    case "I":
+      decodeObj.ans.push("item:" + this._decodeRoute_number2id(nxt));
+      break;
+    case "u":
+      decodeObj.ans.push("unEquip:" + nxt);
+      break;
+    case "e":
+      decodeObj.ans.push("equip:" + this._decodeRoute_number2id(nxt));
+      break;
+    case "s":
+      decodeObj.ans.push("saveEquip:" + nxt);
+      break;
+    case "l":
+      decodeObj.ans.push("loadEquip:" + nxt);
+      break;
+    case "F":
+      decodeObj.ans.push("fly:" + nxt);
+      break;
+    case "c":
+      decodeObj.ans.push("choices:none");
+      break;
+    case "C":
+      decodeObj.ans.push("choices:" + nxt);
+      break;
+    case "S":
+      decodeObj.ans.push("shop:" + nxt);
+      break;
+    case "T":
+      decodeObj.ans.push("turn");
+      break;
+    case "t":
+      decodeObj.ans.push("turn:" + mp[nxt]);
+      break;
+    case "G":
+      decodeObj.ans.push("getNext");
+      break;
+    case "p":
+      decodeObj.ans.push("input:none");
+      break;
+    case "P":
+      decodeObj.ans.push("input:" + nxt);
+      break;
+    case "Q":
+      decodeObj.ans.push("input2:" + nxt);
+      break;
+    case "N":
+      decodeObj.ans.push("no");
+      break;
+    case "M":
+      ++decodeObj.index;
+      decodeObj.ans.push(
+        "move:" + nxt + ":" + this._decodeRoute_getNumber(decodeObj)
+      );
+      break;
+    case "K":
+      decodeObj.ans.push("key:" + nxt);
+      break;
+    case "k":
+      ++decodeObj.index;
+      var px = this._decodeRoute_getNumber(decodeObj);
+      ++decodeObj.index;
+      var py = this._decodeRoute_getNumber(decodeObj);
+      decodeObj.ans.push("click:" + nxt + ":" + px + ":" + py);
+      break;
+    case "X":
+      decodeObj.ans.push("random:" + nxt);
+      break;
+  }
+};
 
 ////// 判断某对象是否不为null也不为NaN //////
 utils.prototype.isset = function (val) {
-    return val != null && !(typeof val == 'number' && isNaN(val));
-}
+  return val != null && !(typeof val == "number" && isNaN(val));
+};
 
 ////// 获得子数组 //////
 utils.prototype.subarray = function (a, b) {
-    if (!(a instanceof Array) || !(b instanceof Array) || a.length < b.length)
-        return null;
-    for (var i = 0; i < b.length; ++i) {
-        if (a[i] != b[i]) return null;
-    }
-    return a.slice(b.length);
-}
+  if (!(a instanceof Array) || !(b instanceof Array) || a.length < b.length)
+    return null;
+  for (var i = 0; i < b.length; ++i) {
+    if (a[i] != b[i]) return null;
+  }
+  return a.slice(b.length);
+};
 
 utils.prototype.inArray = function (array, element) {
-    return (array instanceof Array) && array.indexOf(element) >= 0;
-}
+  return array instanceof Array && array.indexOf(element) >= 0;
+};
 
 utils.prototype.clamp = function (x, a, b) {
-    var min = Math.min(a, b), max = Math.max(a, b);
-    return Math.min(Math.max(x || 0, min), max);
-}
+  var min = Math.min(a, b),
+    max = Math.max(a, b);
+  return Math.min(Math.max(x || 0, min), max);
+};
 
 utils.prototype.getCookie = function (name) {
-    var match = document.cookie.match(new RegExp('(^| )' + name + '=([^;]+)'));
-    return match ? match[2] : null;
-}
+  var match = document.cookie.match(new RegExp("(^| )" + name + "=([^;]+)"));
+  return match ? match[2] : null;
+};
 
 ////// 设置statusBar的innerHTML,会自动斜体和放缩,也可以增加自定义css //////
 utils.prototype.setStatusBarInnerHTML = function (name, value, css) {
-    if (!core.statusBar[name]) return;
-    if (typeof value == 'number') value = this.formatBigNumber(value);
-    var italic = /^[-a-zA-Z0-9`~!@#$%^&*()_=+\[{\]}\\|;:'",<.>\/?]*$/.test(value);
-    var style = 'font-style: ' + (italic ? 'italic' : 'normal') + '; ';
-    style += 'text-shadow: #000 1px 0 0, #000 0 1px 0, #000 -1px 0 0, #000 0 -1px 0; ';
-    // 判定是否需要缩放
-    var length = this.strlen(value) || 1;
-    style += 'font-size: ' + Math.min(1, 7 / length) + 'em; ';
-    if (css) style += css;
-    var _style = core.statusBar[name].getAttribute('_style');
-    var _value = core.statusBar[name].getAttribute('_value');
-    if (_style == style) {
-        if (value == _value) return;
-        core.statusBar[name].children[0].innerText = value;
-    } else {
-        core.statusBar[name].innerHTML = "<span class='_status' style='" + style + "'></span>";
-        core.statusBar[name].children[0].innerText = value;
-        core.statusBar[name].setAttribute('_style', style);
-    }
-    core.statusBar[name].setAttribute('_value', value);;
-}
+  if (!core.statusBar[name]) return;
+  if (typeof value == "number") value = this.formatBigNumber(value);
+  var italic = /^[-a-zA-Z0-9`~!@#$%^&*()_=+\[{\]}\\|;:'",<.>\/?]*$/.test(value);
+  var style = "font-style: " + (italic ? "italic" : "normal") + "; ";
+  style +=
+    "text-shadow: #000 1px 0 0, #000 0 1px 0, #000 -1px 0 0, #000 0 -1px 0; ";
+  // 判定是否需要缩放
+  var length = this.strlen(value) || 1;
+  style += "font-size: " + Math.min(1, 7 / length) + "em; ";
+  if (css) style += css;
+  var _style = core.statusBar[name].getAttribute("_style");
+  var _value = core.statusBar[name].getAttribute("_value");
+  if (_style == style) {
+    if (value == _value) return;
+    core.statusBar[name].children[0].innerText = value;
+  } else {
+    core.statusBar[name].innerHTML =
+      "<span class='_status' style='" + style + "'></span>";
+    core.statusBar[name].children[0].innerText = value;
+    core.statusBar[name].setAttribute("_style", style);
+  }
+  core.statusBar[name].setAttribute("_value", value);
+};
 
 utils.prototype.strlen = function (str) {
-    var count = 0;
-    for (var i = 0, len = str.length; i < len; i++) {
-        count += str.charCodeAt(i) < 256 ? 1 : 2;
-    }
-    return count;
+  var count = 0;
+  for (var i = 0, len = str.length; i < len; i++) {
+    count += str.charCodeAt(i) < 256 ? 1 : 2;
+  }
+  return count;
 };
 
 utils.prototype.turnDirection = function (turn, direction) {
-    direction = direction || core.getHeroLoc('direction');
-    var directionList = ["left", "leftup", "up", "rightup", "right", "rightdown", "down", "leftdown"];
-    if (directionList.indexOf(turn) >= 0) return turn;
-    if (turn == ':hero') return core.getHeroLoc('direction');
-    if (turn == ':backhero') return this.turnDirection(':back', core.getHeroLoc('direction'));
-    if (typeof turn === 'number' && turn % 45 == 0) turn /= 45;
-    else {
-        switch (turn) {
-            case ':left': turn = 6; break; // turn left
-            case ':right': turn = 2; break; // turn right
-            case ':back': turn = 4; break; // turn back
-            default: turn = 0; break;
-        }
+  direction = direction || core.getHeroLoc("direction");
+  var directionList = [
+    "left",
+    "leftup",
+    "up",
+    "rightup",
+    "right",
+    "rightdown",
+    "down",
+    "leftdown",
+  ];
+  if (directionList.indexOf(turn) >= 0) return turn;
+  if (turn == ":hero") return core.getHeroLoc("direction");
+  if (turn == ":backhero")
+    return this.turnDirection(":back", core.getHeroLoc("direction"));
+  if (typeof turn === "number" && turn % 45 == 0) turn /= 45;
+  else {
+    switch (turn) {
+      case ":left":
+        turn = 6;
+        break; // turn left
+      case ":right":
+        turn = 2;
+        break; // turn right
+      case ":back":
+        turn = 4;
+        break; // turn back
+      default:
+        turn = 0;
+        break;
     }
-    var index = directionList.indexOf(direction);
-    if (index < 0) return direction;
-    return directionList[(index + (turn || 0)) % directionList.length];
-}
+  }
+  var index = directionList.indexOf(direction);
+  if (index < 0) return direction;
+  return directionList[(index + (turn || 0)) % directionList.length];
+};
 
 utils.prototype.matchWildcard = function (pattern, string) {
-    try {
-        return new RegExp('^' + pattern.split(/\*+/).map(function (s) {
-            return s.replace(/[|\\{}()[\]^$+*?.]/g, '\\$&');
-        }).join('.*') + '$').test(string);
-    } catch (e) {
-        return false;
-    }
-}
+  try {
+    return new RegExp(
+      "^" +
+        pattern
+          .split(/\*+/)
+          .map(function (s) {
+            return s.replace(/[|\\{}()[\]^$+*?.]/g, "\\$&");
+          })
+          .join(".*") +
+        "$"
+    ).test(string);
+  } catch (e) {
+    return false;
+  }
+};
 
 utils.prototype.matchRegex = function (pattern, string) {
-    try {
-        if (pattern.startsWith("^")) pattern = pattern.substring(1);
-        if (pattern.endsWith("$")) pattern = pattern.substring(0, pattern.length - 1);
-        return new RegExp("^" + pattern + "$").test(string);
-    } catch (e) {
-        return false;
-    }
-}
+  try {
+    if (pattern.startsWith("^")) pattern = pattern.substring(1);
+    if (pattern.endsWith("$"))
+      pattern = pattern.substring(0, pattern.length - 1);
+    return new RegExp("^" + pattern + "$").test(string);
+  } catch (e) {
+    return false;
+  }
+};
 
 ////// Base64加密 //////
 utils.prototype.encodeBase64 = function (str) {
-    return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g, function (match, p1) {
-        return String.fromCharCode(parseInt(p1, 16))
-    }))
-}
+  return btoa(
+    encodeURIComponent(str).replace(/%([0-9A-F]{2})/g, function (match, p1) {
+      return String.fromCharCode(parseInt(p1, 16));
+    })
+  );
+};
 
 ////// Base64解密 //////
 utils.prototype.decodeBase64 = function (str) {
-    return decodeURIComponent(atob(str).split('').map(function (c) {
-        return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2);
-    }).join(''));
-}
+  return decodeURIComponent(
+    atob(str)
+      .split("")
+      .map(function (c) {
+        return "%" + ("00" + c.charCodeAt(0).toString(16)).slice(-2);
+      })
+      .join("")
+  );
+};
 
 utils.prototype.rand = function (num) {
-    var rand = core.getFlag('__rand__');
-    rand = this.__next_rand(rand);
-    core.setFlag('__rand__', rand);
-    var ans = rand / 2147483647;
-    if (num && num > 0)
-        return Math.floor(ans * num);
-    return ans;
-}
+  var rand = core.getFlag("__rand__");
+  rand = this.__next_rand(rand);
+  core.setFlag("__rand__", rand);
+  var ans = rand / 2147483647;
+  if (num && num > 0) return Math.floor(ans * num);
+  return ans;
+};
 
 ////// 生成随机数(录像方法) //////
 utils.prototype.rand2 = function (num) {
-    num = num || 2147483648;
-    num = Math.abs(num);
+  num = num || 2147483648;
+  num = Math.abs(num);
 
-    var value;
-    if (core.isReplaying()) {
-        var action = core.status.replay.toReplay.shift();
-        if (action.indexOf("random:") == 0) {
-            value = parseInt(action.substring(7));
-            if (isNaN(value) || value >= num || value < 0) {
-                console.warn('错误!当前random:项超过范围。将重新随机生成!');
-                value = Math.floor(Math.random() * num);
-            }
-        }
-        else {
-            console.warn('错误!当前需要一个random:项。将重新随机生成!');
-            value = Math.floor(Math.random() * num);
-        }
-    }
-    else {
+  var value;
+  if (core.isReplaying()) {
+    var action = core.status.replay.toReplay.shift();
+    if (action.indexOf("random:") == 0) {
+      value = parseInt(action.substring(7));
+      if (isNaN(value) || value >= num || value < 0) {
+        console.warn("错误!当前random:项超过范围。将重新随机生成!");
         value = Math.floor(Math.random() * num);
+      }
+    } else {
+      console.warn("错误!当前需要一个random:项。将重新随机生成!");
+      value = Math.floor(Math.random() * num);
     }
-    core.status.route.push("random:" + value);
-    return value;
-}
+  } else {
+    value = Math.floor(Math.random() * num);
+  }
+  core.status.route.push("random:" + value);
+  return value;
+};
 
 utils.prototype.__init_seed = function () {
-    var rand = new Date().getTime() % 34834795 + 3534;
-    rand = this.__next_rand(rand);
-    rand = this.__next_rand(rand);
-    rand = this.__next_rand(rand);
-    core.setFlag('__seed__', rand);
-    core.setFlag('__rand__', rand);
-}
+  var rand = (new Date().getTime() % 34834795) + 3534;
+  rand = this.__next_rand(rand);
+  rand = this.__next_rand(rand);
+  rand = this.__next_rand(rand);
+  core.setFlag("__seed__", rand);
+  core.setFlag("__rand__", rand);
+};
 
 utils.prototype.__next_rand = function (_rand) {
-    _rand = (_rand % 127773) * 16807 - ~~(_rand / 127773) * 2836;
-    _rand += _rand < 0 ? 2147483647 : 0;
-    return _rand;
-}
+  _rand = (_rand % 127773) * 16807 - ~~(_rand / 127773) * 2836;
+  _rand += _rand < 0 ? 2147483647 : 0;
+  return _rand;
+};
 
 ////// 读取一个本地文件内容 //////
 utils.prototype.readFile = function (success, error, accept, readType) {
+  core.platform.successCallback = success;
+  core.platform.errorCallback = error;
 
-    core.platform.successCallback = success;
-    core.platform.errorCallback = error;
+  if (window.jsinterface) {
+    window.jsinterface.readFile();
+    return;
+  }
 
-    if (window.jsinterface) {
-        window.jsinterface.readFile();
+  // step 0: 不为http/https,直接不支持
+  if (!core.platform.isOnline) {
+    alert("离线状态下不支持文件读取!");
+    if (error) error();
+    return;
+  }
+
+  // Step 1: 如果不支持FileReader,直接不支持
+  if (core.platform.fileReader == null) {
+    alert("当前浏览器不支持FileReader!");
+    if (error) error();
+    return;
+  }
+
+  if (core.platform.fileInput == null) {
+    core.platform.fileInput = document.createElement("input");
+    core.platform.fileInput.style.opacity = 0;
+    core.platform.fileInput.type = "file";
+    core.platform.fileInput.onchange = function () {
+      var files = core.platform.fileInput.files;
+      if (files.length == 0) {
+        if (core.platform.errorCallback) core.platform.errorCallback();
         return;
-    }
+      }
+      if (!readType)
+        core.platform.fileReader.readAsText(core.platform.fileInput.files[0]);
+      else
+        core.platform.fileReader.readAsDataURL(
+          core.platform.fileInput.files[0]
+        );
+      core.platform.fileInput.value = "";
+    };
+  }
+  core.platform.fileInput.value = "";
+  if (accept) core.platform.fileInput.accept = accept;
 
-    // step 0: 不为http/https,直接不支持
-    if (!core.platform.isOnline) {
-        alert("离线状态下不支持文件读取!");
-        if (error) error();
-        return;
-    }
-
-    // Step 1: 如果不支持FileReader,直接不支持
-    if (core.platform.fileReader == null) {
-        alert("当前浏览器不支持FileReader!");
-        if (error) error();
-        return;
-    }
-
-    if (core.platform.fileInput == null) {
-        core.platform.fileInput = document.createElement("input");
-        core.platform.fileInput.style.opacity = 0;
-        core.platform.fileInput.type = 'file';
-        core.platform.fileInput.onchange = function () {
-            var files = core.platform.fileInput.files;
-            if (files.length == 0) {
-                if (core.platform.errorCallback)
-                    core.platform.errorCallback();
-                return;
-            }
-            if (!readType) core.platform.fileReader.readAsText(core.platform.fileInput.files[0]);
-            else core.platform.fileReader.readAsDataURL(core.platform.fileInput.files[0]);
-            core.platform.fileInput.value = '';
-        }
-    }
-    core.platform.fileInput.value = '';
-    if (accept) core.platform.fileInput.accept = accept;
-
-    core.platform.fileInput.click();
-}
+  core.platform.fileInput.click();
+};
 
 ////// 读取文件完毕 //////
 utils.prototype.readFileContent = function (content) {
-    var obj = null;
-    if (content.slice(0, 4) === 'data') {
-        if (core.platform.successCallback)
-            core.platform.successCallback(content);
-        return;
-    }
-    // 检查base64
+  var obj = null;
+  if (content.slice(0, 4) === "data") {
+    if (core.platform.successCallback) core.platform.successCallback(content);
+    return;
+  }
+  // 检查base64
+  try {
+    obj = JSON.parse(LZString.decompressFromBase64(content));
+  } catch (e) {}
+  if (!obj) {
     try {
-        obj = JSON.parse(LZString.decompressFromBase64(content));
-    } catch (e) { }
-    if (!obj) {
-        try {
-            obj = JSON.parse(content);
-        } catch (e) {
-            console.error(e)
-        }
+      obj = JSON.parse(content);
+    } catch (e) {
+      console.error(e);
     }
+  }
 
-    if (obj) {
-        if (core.platform.successCallback)
-            core.platform.successCallback(obj);
-        return;
-    }
+  if (obj) {
+    if (core.platform.successCallback) core.platform.successCallback(obj);
+    return;
+  }
 
-    if (core.platform.errorCallback)
-        core.platform.errorCallback();
-}
+  if (core.platform.errorCallback) core.platform.errorCallback();
+};
 
 ////// 下载文件到本地 //////
 utils.prototype.download = function (filename, content) {
+  if (window.jsinterface) {
+    window.jsinterface.download(filename, content);
+    return;
+  }
 
-    if (window.jsinterface) {
-        window.jsinterface.download(filename, content);
-        return;
-    }
+  // Step 0: 不为http/https,直接不支持
+  if (!core.platform.isOnline) {
+    alert("离线状态下不支持下载操作!");
+    return;
+  }
 
-    // Step 0: 不为http/https,直接不支持
-    if (!core.platform.isOnline) {
-        alert("离线状态下不支持下载操作!");
-        return;
+  // Step 1: 如果是iOS平台,直接不支持
+  if (core.platform.isIOS) {
+    if (core.copy(content)) {
+      alert(
+        "iOS平台下不支持直接下载文件!\n所有应下载内容已经复制到您的剪切板,请自行创建空白文件并粘贴。"
+      );
+    } else {
+      alert("iOS平台下不支持下载操作!");
     }
+    return;
+  }
 
-    // Step 1: 如果是iOS平台,直接不支持
-    if (core.platform.isIOS) {
-        if (core.copy(content)) {
-            alert("iOS平台下不支持直接下载文件!\n所有应下载内容已经复制到您的剪切板,请自行创建空白文件并粘贴。");
-        }
-        else {
-            alert("iOS平台下不支持下载操作!");
-        }
-        return;
+  // Step 2: 如果不是PC平台(Android),则只支持chrome
+  if (!core.platform.isPC) {
+    if (
+      !core.platform.isChrome ||
+      core.platform.isQQ ||
+      core.platform.isWeChat
+    ) {
+      // 检测chrome
+      if (core.copy(content)) {
+        alert(
+          "移动端只有Chrome浏览器支持直接下载文件!\n所有应下载内容已经复制到您的剪切板,请自行创建空白文件并粘贴。"
+        );
+      } else {
+        alert("该平台或浏览器暂不支持下载操作!");
+      }
+      return;
     }
+  }
 
-    // Step 2: 如果不是PC平台(Android),则只支持chrome
-    if (!core.platform.isPC) {
-        if (!core.platform.isChrome || core.platform.isQQ || core.platform.isWeChat) { // 检测chrome
-            if (core.copy(content)) {
-                alert("移动端只有Chrome浏览器支持直接下载文件!\n所有应下载内容已经复制到您的剪切板,请自行创建空白文件并粘贴。");
-            }
-            else {
-                alert("该平台或浏览器暂不支持下载操作!");
-            }
-            return;
-        }
-    }
+  // Step 3: 如果是Safari浏览器,则提示并打开新窗口
+  if (core.platform.isSafari) {
+    alert(
+      "你当前使用的是Safari浏览器,不支持直接下载文件。\n即将打开一个新窗口为应下载内容,请自行全选复制然后创建空白文件并粘贴。"
+    );
+    var blob = new Blob([content], { type: "text/plain;charset=utf-8" });
+    var href = window.URL.createObjectURL(blob);
+    var opened = window.open(href, "_blank");
+    window.URL.revokeObjectURL(href);
+    return;
+  }
 
-    // Step 3: 如果是Safari浏览器,则提示并打开新窗口
-    if (core.platform.isSafari) {
-        alert("你当前使用的是Safari浏览器,不支持直接下载文件。\n即将打开一个新窗口为应下载内容,请自行全选复制然后创建空白文件并粘贴。");
-        var blob = new Blob([content], { type: 'text/plain;charset=utf-8' });
-        var href = window.URL.createObjectURL(blob);
-        var opened = window.open(href, "_blank");
-        window.URL.revokeObjectURL(href);
-        return;
-    }
-
-    // Step 4: 下载
-    var blob = new Blob([content], { type: 'text/plain;charset=utf-8' });
-    if (window.navigator.msSaveOrOpenBlob) {
-        window.navigator.msSaveBlob(blob, filename);
-    }
-    else {
-        var href = window.URL.createObjectURL(blob);
-        var elem = window.document.createElement('a');
-        elem.href = href;
-        elem.download = filename;
-        document.body.appendChild(elem);
-        elem.click();
-        document.body.removeChild(elem);
-        window.URL.revokeObjectURL(href);
-    }
-}
+  // Step 4: 下载
+  var blob = new Blob([content], { type: "text/plain;charset=utf-8" });
+  if (window.navigator.msSaveOrOpenBlob) {
+    window.navigator.msSaveBlob(blob, filename);
+  } else {
+    var href = window.URL.createObjectURL(blob);
+    var elem = window.document.createElement("a");
+    elem.href = href;
+    elem.download = filename;
+    document.body.appendChild(elem);
+    elem.click();
+    document.body.removeChild(elem);
+    window.URL.revokeObjectURL(href);
+  }
+};
 
 ////// 复制一段内容到剪切板 //////
 utils.prototype.copy = function (data) {
+  if (window.jsinterface) {
+    window.jsinterface.copy(data);
+    return true;
+  }
 
-    if (window.jsinterface) {
-        window.jsinterface.copy(data);
-        return true;
-    }
+  if (!core.platform.supportCopy) return false;
 
-    if (!core.platform.supportCopy) return false;
+  var textArea = document.createElement("textarea");
+  textArea.style.position = "fixed";
+  textArea.style.top = 0;
+  textArea.style.left = 0;
+  textArea.style.width = "2em";
+  textArea.style.height = "2em";
+  textArea.style.padding = 0;
+  textArea.style.border = "none";
+  textArea.style.outline = "none";
+  textArea.style.boxShadow = "none";
+  textArea.style.background = "transparent";
+  textArea.value = data;
+  document.body.appendChild(textArea);
+  textArea.focus();
+  textArea.setSelectionRange(0, textArea.value.length);
+  var successful = false;
+  try {
+    successful = document.execCommand("copy");
+  } catch (err) {
+    successful = false;
+  }
 
-    var textArea = document.createElement("textarea");
-    textArea.style.position = 'fixed';
-    textArea.style.top = 0;
-    textArea.style.left = 0;
-    textArea.style.width = '2em';
-    textArea.style.height = '2em';
-    textArea.style.padding = 0;
-    textArea.style.border = 'none';
-    textArea.style.outline = 'none';
-    textArea.style.boxShadow = 'none';
-    textArea.style.background = 'transparent';
-    textArea.value = data;
-    document.body.appendChild(textArea);
-    textArea.focus();
-    textArea.setSelectionRange(0, textArea.value.length);
-    var successful = false;
-    try {
-        successful = document.execCommand('copy');
-    } catch (err) {
-        successful = false;
-    }
-
-    document.body.removeChild(textArea);
-    return successful;
-}
+  document.body.removeChild(textArea);
+  return successful;
+};
 
 ////// 显示一段confirm //////
 utils.prototype.myconfirm = function (hint, yesCallback, noCallback) {
-    main.dom.inputDiv.style.display = 'block';
-    main.dom.inputMessage.innerHTML = hint.replace(/\n/g, '<br/>');
-    main.dom.inputBox.style.display = 'none';
-    main.dom.inputYes.blur();
-    main.dom.inputNo.blur();
-    core.status.holdingKeys = [];
+  main.dom.inputDiv.style.display = "block";
+  main.dom.inputMessage.innerHTML = hint.replace(/\n/g, "<br/>");
+  main.dom.inputBox.style.display = "none";
+  main.dom.inputYes.blur();
+  main.dom.inputNo.blur();
+  core.status.holdingKeys = [];
 
-    core.platform.successCallback = yesCallback;
-    core.platform.errorCallback = noCallback;
-}
+  core.platform.successCallback = yesCallback;
+  core.platform.errorCallback = noCallback;
+};
 
 ////// 让用户输入一段文字 //////
 utils.prototype.myprompt = function (hint, value, callback) {
-    main.dom.inputDiv.style.display = 'block';
-    main.dom.inputMessage.innerHTML = hint.replace(/\n/g, '<br/>');
-    main.dom.inputBox.style.display = 'block';
-    main.dom.inputBox.value = value == null ? "" : value;
-    main.dom.inputYes.blur();
-    main.dom.inputNo.blur();
-    setTimeout(function () {
-        main.dom.inputBox.focus();
-    });
-    core.status.holdingKeys = [];
+  main.dom.inputDiv.style.display = "block";
+  main.dom.inputMessage.innerHTML = hint.replace(/\n/g, "<br/>");
+  main.dom.inputBox.style.display = "block";
+  main.dom.inputBox.value = value == null ? "" : value;
+  main.dom.inputYes.blur();
+  main.dom.inputNo.blur();
+  setTimeout(function () {
+    main.dom.inputBox.focus();
+  });
+  core.status.holdingKeys = [];
 
-    core.platform.successCallback = core.platform.errorCallback = callback;
-}
+  core.platform.successCallback = core.platform.errorCallback = callback;
+};
 
 ////// 动画显示某对象 //////
 utils.prototype.showWithAnimate = function (obj, speed, callback) {
-    obj.style.display = 'block';
-    if (!speed || main.mode != 'play') {
-        obj.style.opacity = 1;
-        if (callback) callback();
-        return;
+  obj.style.display = "block";
+  if (!speed || main.mode != "play") {
+    obj.style.opacity = 1;
+    if (callback) callback();
+    return;
+  }
+  obj.style.opacity = 0;
+  var opacityVal = 0;
+  var showAnimate = window.setInterval(function () {
+    opacityVal += 0.03;
+    obj.style.opacity = opacityVal;
+    if (opacityVal > 1) {
+      clearInterval(showAnimate);
+      if (callback) callback();
     }
-    obj.style.opacity = 0;
-    var opacityVal = 0;
-    var showAnimate = window.setInterval(function () {
-        opacityVal += 0.03;
-        obj.style.opacity = opacityVal;
-        if (opacityVal > 1) {
-            clearInterval(showAnimate);
-            if (callback) callback();
-        }
-    }, speed);
-}
+  }, speed);
+};
 
 ////// 动画使某对象消失 //////
 utils.prototype.hideWithAnimate = function (obj, speed, callback) {
-    if (!speed || main.mode != 'play') {
-        obj.style.display = 'none';
-        if (callback) callback();
-        return;
+  if (!speed || main.mode != "play") {
+    obj.style.display = "none";
+    if (callback) callback();
+    return;
+  }
+  obj.style.opacity = 1;
+  var opacityVal = 1;
+  var hideAnimate = window.setInterval(function () {
+    opacityVal -= 0.03;
+    obj.style.opacity = opacityVal;
+    if (opacityVal < 0) {
+      obj.style.display = "none";
+      clearInterval(hideAnimate);
+      if (callback) callback();
     }
-    obj.style.opacity = 1;
-    var opacityVal = 1;
-    var hideAnimate = window.setInterval(function () {
-        opacityVal -= 0.03;
-        obj.style.opacity = opacityVal;
-        if (opacityVal < 0) {
-            obj.style.display = 'none';
-            clearInterval(hideAnimate);
-            if (callback) callback();
-        }
-    }, speed);
-}
+  }, speed);
+};
 
 ////// 生成浏览器唯一的 guid //////
 utils.prototype.getGuid = function () {
-    var guid = localStorage.getItem('guid');
-    if (guid != null) return guid;
-    guid = 'xxxxxxxx_xxxx_4xxx_yxxx_xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
-        var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
-        return v.toString(16);
-    });
-    localStorage.setItem('guid', guid);
-    return guid;
-}
+  var guid = localStorage.getItem("guid");
+  if (guid != null) return guid;
+  guid = "xxxxxxxx_xxxx_4xxx_yxxx_xxxxxxxxxxxx".replace(/[xy]/g, function (c) {
+    var r = (Math.random() * 16) | 0,
+      v = c == "x" ? r : (r & 0x3) | 0x8;
+    return v.toString(16);
+  });
+  localStorage.setItem("guid", guid);
+  return guid;
+};
 
 utils.prototype.hashCode = function (obj) {
-    if (typeof obj == 'string') {
-        var hash = 0, i, chr;
-        if (obj.length === 0) return hash;
-        for (i = 0; i < obj.length; i++) {
-            chr = obj.charCodeAt(i);
-            hash = ((hash << 5) - hash) + chr;
-            hash |= 0;
-        }
-        return hash;
+  if (typeof obj == "string") {
+    var hash = 0,
+      i,
+      chr;
+    if (obj.length === 0) return hash;
+    for (i = 0; i < obj.length; i++) {
+      chr = obj.charCodeAt(i);
+      hash = (hash << 5) - hash + chr;
+      hash |= 0;
     }
-    return this.hashCode(JSON.stringify(obj).split("").sort().join(""));
-}
+    return hash;
+  }
+  return this.hashCode(JSON.stringify(obj).split("").sort().join(""));
+};
 
 utils.prototype.same = function (a, b) {
-    if (a == null && b == null) return true;
-    if (a == null || b == null) return false;
-    if (a === b) return true;
-    if (a instanceof Array && b instanceof Array) {
-        if (a.length != b.length) return false;
-        for (var i = 0; i < a.length; i++) {
-            if (!this.same(a[i], b[i])) return false;
+  if (a == null && b == null) return true;
+  if (a == null || b == null) return false;
+  if (a === b) return true;
+  if (a instanceof Array && b instanceof Array) {
+    if (a.length != b.length) return false;
+    for (var i = 0; i < a.length; i++) {
+      if (!this.same(a[i], b[i])) return false;
+    }
+    return true;
+  }
+  if (a instanceof Object && b instanceof Object) {
+    var obj = {};
+    for (var i in a) obj[i] = true;
+    for (var i in b) obj[i] = true;
+    for (var i in obj) {
+      if (!this.same(a[i], b[i])) return false;
+    }
+    return true;
+  }
+  return false;
+};
+
+utils.prototype.unzip = function (
+  blobOrUrl,
+  success,
+  error,
+  convertToText,
+  onprogress
+) {
+  var _error = function (msg) {
+    console.error(msg);
+    if (error) error(msg);
+  };
+
+  if (!window.zip) {
+    return _error("zip.js not exists!");
+  }
+
+  if (typeof blobOrUrl == "string") {
+    return core.http(
+      "GET",
+      blobOrUrl,
+      null,
+      function (data) {
+        core.unzip(data, success, error, convertToText);
+      },
+      _error,
+      null,
+      "blob",
+      onprogress
+    );
+  }
+
+  if (!(blobOrUrl instanceof Blob)) {
+    return _error("Should use Blob or URL as input");
+  }
+
+  zip.createReader(
+    new zip.BlobReader(blobOrUrl),
+    function (reader) {
+      reader.getEntries(function (entries) {
+        core.utils._unzip_readEntries(
+          entries,
+          function (data) {
+            reader.close(function () {
+              if (success) success(data);
+            });
+          },
+          convertToText
+        );
+      });
+    },
+    _error
+  );
+};
+
+utils.prototype._unzip_readEntries = function (
+  entries,
+  success,
+  convertToText
+) {
+  var results = {};
+  if (entries == null || entries.length == 0) {
+    return success(results);
+  }
+  var length = entries.length;
+  entries.forEach(function (entry) {
+    entry.getData(
+      convertToText ? new zip.TextWriter("utf8") : new zip.BlobWriter(),
+      function (data) {
+        results[entry.filename] = data;
+        length--;
+        if (length == 0) {
+          success(results);
         }
-        return true;
-    }
-    if (a instanceof Object && b instanceof Object) {
-        var obj = {};
-        for (var i in a) obj[i] = true;
-        for (var i in b) obj[i] = true;
-        for (var i in obj) {
-            if (!this.same(a[i], b[i])) return false;
-        }
-        return true;
-    }
-    return false;
-}
+      }
+    );
+  });
+};
 
-utils.prototype.unzip = function (blobOrUrl, success, error, convertToText, onprogress) {
-    var _error = function (msg) {
-        console.error(msg);
-        if (error) error(msg);
+utils.prototype.http = function (
+  type,
+  url,
+  formData,
+  success,
+  error,
+  mimeType,
+  responseType,
+  onprogress
+) {
+  var xhr = new XMLHttpRequest();
+  xhr.open(type, url, true);
+  if (mimeType) xhr.overrideMimeType(mimeType);
+  if (responseType) xhr.responseType = responseType;
+  xhr.onload = function (e) {
+    if (xhr.status == 200) {
+      if (success) success(xhr.response);
+    } else {
+      if (error) error("HTTP " + xhr.status);
     }
-
-    if (!window.zip) {
-        return _error("zip.js not exists!");
+  };
+  xhr.onprogress = function (e) {
+    if (e.lengthComputable) {
+      if (onprogress) onprogress(e.loaded, e.total);
     }
-
-    if (typeof blobOrUrl == 'string') {
-        return core.http('GET', blobOrUrl, null, function (data) {
-            core.unzip(data, success, error, convertToText);
-        }, _error, null, 'blob', onprogress);
-    }
-
-    if (!(blobOrUrl instanceof Blob)) {
-        return _error("Should use Blob or URL as input");
-    }
-
-    zip.createReader(new zip.BlobReader(blobOrUrl), function (reader) {
-        reader.getEntries(function (entries) {
-            core.utils._unzip_readEntries(entries, function (data) {
-                reader.close(function () {
-                    if (success) success(data);
-                });
-            }, convertToText);
-        });
-    }, _error);
-}
-
-utils.prototype._unzip_readEntries = function (entries, success, convertToText) {
-    var results = {};
-    if (entries == null || entries.length == 0) {
-        return success(results);
-    }
-    var length = entries.length;
-    entries.forEach(function (entry) {
-        entry.getData(convertToText ? new zip.TextWriter('utf8') : new zip.BlobWriter(), function (data) {
-            results[entry.filename] = data;
-            length--;
-            if (length == 0) {
-                success(results);
-            }
-        });
-    });
-}
-
-utils.prototype.http = function (type, url, formData, success, error, mimeType, responseType, onprogress) {
-    var xhr = new XMLHttpRequest();
-    xhr.open(type, url, true);
-    if (mimeType) xhr.overrideMimeType(mimeType);
-    if (responseType) xhr.responseType = responseType;
-    xhr.onload = function (e) {
-        if (xhr.status == 200) {
-            if (success) success(xhr.response);
-        }
-        else {
-            if (error) error("HTTP " + xhr.status);
-        }
-    };
-    xhr.onprogress = function (e) {
-        if (e.lengthComputable) {
-            if (onprogress) onprogress(e.loaded, e.total);
-        }
-    }
-    xhr.onabort = function () {
-        if (error) error("Abort");
-    }
-    xhr.ontimeout = function () {
-        if (error) error("Timeout");
-    }
-    xhr.onerror = function () {
-        if (error) error("Error on Connection");
-    }
-    if (formData)
-        xhr.send(formData);
-    else xhr.send();
-}
+  };
+  xhr.onabort = function () {
+    if (error) error("Abort");
+  };
+  xhr.ontimeout = function () {
+    if (error) error("Timeout");
+  };
+  xhr.onerror = function () {
+    if (error) error("Error on Connection");
+  };
+  if (formData) xhr.send(formData);
+  else xhr.send();
+};
 
 // LZW-compress
 // https://gist.github.com/revolunet/843889
-function lzw_encode (s) {
-    var dict = {};
-    var data = (s + "").split("");
-    var out = [];
-    var currChar;
-    var phrase = data[0];
-    var code = 256;
-    for (var i = 1; i < data.length; i++) {
-        currChar = data[i];
-        if (dict[phrase + currChar] != null) {
-            phrase += currChar;
-        }
-        else {
-            out.push(phrase.length > 1 ? dict[phrase] : phrase.charCodeAt(0));
-            dict[phrase + currChar] = code;
-            code++;
-            phrase = currChar;
-        }
+function lzw_encode(s) {
+  var dict = {};
+  var data = (s + "").split("");
+  var out = [];
+  var currChar;
+  var phrase = data[0];
+  var code = 256;
+  for (var i = 1; i < data.length; i++) {
+    currChar = data[i];
+    if (dict[phrase + currChar] != null) {
+      phrase += currChar;
+    } else {
+      out.push(phrase.length > 1 ? dict[phrase] : phrase.charCodeAt(0));
+      dict[phrase + currChar] = code;
+      code++;
+      phrase = currChar;
     }
-    out.push(phrase.length > 1 ? dict[phrase] : phrase.charCodeAt(0));
-    for (var i = 0; i < out.length; i++) {
-        out[i] = String.fromCharCode(out[i]);
-    }
-    return out.join("");
+  }
+  out.push(phrase.length > 1 ? dict[phrase] : phrase.charCodeAt(0));
+  for (var i = 0; i < out.length; i++) {
+    out[i] = String.fromCharCode(out[i]);
+  }
+  return out.join("");
 }
 
 // Decompress an LZW-encoded string
-function lzw_decode (s) {
-    var dict = {};
-    var data = (s + "").split("");
-    var currChar = data[0];
-    var oldPhrase = currChar;
-    var out = [currChar];
-    var code = 256;
-    var phrase;
-    for (var i = 1; i < data.length; i++) {
-        var currCode = data[i].charCodeAt(0);
-        if (currCode < 256) {
-            phrase = data[i];
-        }
-        else {
-            phrase = dict[currCode] ? dict[currCode] : (oldPhrase + currChar);
-        }
-        out.push(phrase);
-        currChar = phrase.charAt(0);
-        dict[code] = oldPhrase + currChar;
-        code++;
-        oldPhrase = phrase;
+function lzw_decode(s) {
+  var dict = {};
+  var data = (s + "").split("");
+  var currChar = data[0];
+  var oldPhrase = currChar;
+  var out = [currChar];
+  var code = 256;
+  var phrase;
+  for (var i = 1; i < data.length; i++) {
+    var currCode = data[i].charCodeAt(0);
+    if (currCode < 256) {
+      phrase = data[i];
+    } else {
+      phrase = dict[currCode] ? dict[currCode] : oldPhrase + currChar;
     }
-    return out.join("");
+    out.push(phrase);
+    currChar = phrase.charAt(0);
+    dict[code] = oldPhrase + currChar;
+    code++;
+    oldPhrase = phrase;
+  }
+  return out.join("");
 }
diff --git a/main.js b/main.js
index a9242ca..0295c32 100644
--- a/main.js
+++ b/main.js
@@ -276,7 +276,7 @@ main.prototype.init = function (mode, callback) {
           coreData[t] = main[t];
         });
         main.core.init(coreData, callback);
-        main.core.resize();
+        core.resize();
         // 自动放缩最大化
         if (!main.replayChecking) {
           if (core.getLocalStorage("autoScale") == null) {
@@ -492,7 +492,6 @@ main.prototype.selectButton = function (index) {
 main.prototype.importFonts = function (fonts) {
   if (!(fonts instanceof Array) || fonts.length == 0) return;
   var style = document.createElement("style");
-  style.type = "text/css";
   var html = "";
   fonts.forEach(function (font) {
     html +=
@@ -910,7 +909,7 @@ main.prototype.listen = function () {
   ////// 点击“开始游戏”时 //////
   main.dom.playGame.onclick = function () {
     main.dom.startButtons.style.display = "none";
-    main.core.control.checkBgm();
+    main.core.checkBgm();
 
     if (main.levelChoose.length == 0) {
       core.events.startGame("");
@@ -923,13 +922,13 @@ main.prototype.listen = function () {
 
   ////// 点击“载入游戏”时 //////
   main.dom.loadGame.onclick = function () {
-    main.core.control.checkBgm();
+    main.core.checkBgm();
     main.core.load();
   };
 
   ////// 点击“录像回放”时 //////
   main.dom.replayGame.onclick = function () {
-    main.core.control.checkBgm();
+    main.core.checkBgm();
     main.core.chooseReplayFile();
   };
 
diff --git a/project/animates/hand.animate b/project/animates/hand.animate
index 09c9514..936d7d4 100644
--- a/project/animates/hand.animate
+++ b/project/animates/hand.animate
@@ -1 +1 @@
-{"ratio":2,"se":"attack.mp3","bitmaps":["","","","","","","","","",""],"frame_max":8,"frames":[[[0,0,0,30,120]],[[0,0,0,50,255]],[[0,0,0,70,255]],[[1,0,0,80,255]],[[1,0,0,90,255]],[[2,0,0,90,120]],[],[]]}
\ No newline at end of file
+{"ratio": 2.0, "bitmaps": ["", "", "", "", "", "", "", "", "", "", "", "", "", "", "", ""], "frame_max": 16, "frames": [[[0, 0, 0, 30, 120, 0, 0]], [[0, 0, 0, 35, 153, 0, 0]], [[0, 0, 0, 40, 187, 0, 0]], [[0, 0, 0, 45, 221, 0, 0]], [[0, 0, 0, 50, 255, 0, 0]], [[0, 0, 0, 55, 255, 0, 0]], [[0, 0, 0, 60, 255, 0, 0]], [[0, 0, 0, 65, 255, 0, 0]], [[0, 0, 0, 70, 255, 0, 0]], [[0, 0, 0, 72, 255, 0, 0]], [[0, 0, 0, 75, 255, 0, 0]], [[0, 0, 0, 77, 255, 0, 0]], [[1, 0, 0, 90, 255, 0, 0]], [[1, 0, 0, 90, 255, 0, 0]], [[1, 0, 0, 90, 170, 0, 0]], [[1, 0, 0, 90, 120, 0, 0]]], "se": {"1": "attack.opus"}, "pitch": {"1": 100}}
\ No newline at end of file
diff --git a/project/animates/jianji.animate b/project/animates/jianji.animate
index 2720aaf..fb9fe34 100644
--- a/project/animates/jianji.animate
+++ b/project/animates/jianji.animate
@@ -1 +1 @@
-{"ratio":2,"bitmaps":["","","","","","","","","","","","","","","","","","","",""],"frame_max":20,"frames":[[[5,0,0,50,255,0,120]],[[6,0,0,50,255,0,120]],[[7,0,0,50,255,0,120]],[[8,0,0,50,255,0,120]],[[5,0,0,60,255,1,60]],[[5,0,0,60,255,1,60]],[[5,0,0,60,255,1,60]],[[6,0,0,60,255,1,60]],[[7,0,0,60,255,1,60]],[[8,0,0,60,200,1,60]],[[5,-8,-8,60,255,0,60]],[[6,0,-8,60,200,0,60]],[[6,0,-8,60,255,0,60]],[[6,0,-8,60,255,0,60]],[[7,0,-8,60,255,0,60]],[[8,0,-8,60,255,0,60]],[[8,0,-8,60,255,0,60]],[[9,0,-8,60,255,0,60]],[[9,0,-8,60,200,0,60]],[[9,0,-8,60,50,0,60]]]}
\ No newline at end of file
+{"ratio": 2.0, "bitmaps": ["", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", ""], "frame_max": 17, "frames": [[[0, 0, 0, 60, 255, 1, 60]], [[1, 0, 0, 60, 255, 1, 60]], [[2, 0, 0, 60, 255, 1, 60]], [[3, 0, 0, 60, 255, 1, 60]], [[4, 0, 0, 60, 255, 1, 60]], [[5, 0, 0, 60, 255, 1, 60]], [[6, 0, 0, 60, 255, 1, 60]], [[7, 0, 0, 60, 255, 1, 60]], [[8, 0, 0, 60, 255, 1, 60]], [[9, 0, 0, 60, 255, 1, 60]], [[10, 0, 0, 60, 255, 1, 60]], [[11, 0, 0, 60, 255, 1, 60]], [[12, 0, 0, 60, 255, 1, 60]], [[13, 0, 0, 60, 241, 1, 60]], [[14, 0, 0, 60, 227, 1, 60]], [[15, 0, 0, 60, 213, 1, 60]], [[16, 0, 0, 60, 200, 1, 60]]]}
\ No newline at end of file
diff --git a/project/animates/jianji2.animate b/project/animates/jianji2.animate
new file mode 100644
index 0000000..4991c6f
--- /dev/null
+++ b/project/animates/jianji2.animate
@@ -0,0 +1 @@
+{"ratio": 1.0, "bitmaps": ["", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", ""], "frame_max": 21, "frames": [[[0, 0, 0, 100, 255, 0, 0]], [[1, 0, 0, 100, 255, 0, 0]], [[2, 0, 0, 100, 255, 0, 0]], [[3, 0, 0, 100, 255, 0, 0]], [[4, 0, 0, 100, 255, 0, 0]], [[5, 0, 0, 100, 255, 0, 0]], [[6, 0, 0, 100, 255, 0, 0]], [[7, 0, 0, 100, 255, 0, 0]], [[8, 0, 0, 100, 255, 0, 0]], [[9, 0, 0, 100, 255, 0, 0]], [[10, 0, 0, 100, 255, 0, 0]], [[11, 0, 0, 100, 255, 0, 0]], [[12, 0, 0, 100, 255, 0, 0]], [[13, 0, 0, 100, 255, 0, 0]], [[14, 0, 0, 100, 255, 0, 0]], [[15, 0, 0, 100, 255, 0, 0]], [[16, 0, 0, 100, 255, 0, 0]], [[17, 0, 0, 100, 255, 0, 0]], [[18, 0, 0, 100, 255, 0, 0]], [[19, 0, 0, 100, 255, 0, 0]], [[20, 0, 0, 100, 255, 0, 0]]]}
\ No newline at end of file
diff --git a/project/animates/sword.animate b/project/animates/sword.animate
index 21cc63a..26a8fc2 100644
--- a/project/animates/sword.animate
+++ b/project/animates/sword.animate
@@ -1 +1 @@
-{"ratio":2,"bitmaps":["","","","","","","","","","","","","","","","","","","",""],"frame_max":5,"frames":[[[0,8,32,30,100]],[[0,-16,32,50,130]],[[0,-24,40,80,150]],[[0,-48,24,90,180],[1,-8,40,100,255]],[[0,-56,24,100,200],[1,-8,24,100,255]]]}
\ No newline at end of file
+{"ratio":2,"bitmaps":["","","","","","","","","","","","","","","","","","","",""],"frame_max":5,"frames":[[[0,8,32,30,100]],[[0,-16,32,50,130]],[[0,-24,40,80,150]],[[0,-48,24,90,180],[1,-8,40,100,255]],[[0,-56,24,100,200],[1,-8,24,100,255]]],"se":{},"pitch":{}}
\ No newline at end of file
diff --git a/project/animates/thunder.animate b/project/animates/thunder.animate
index f5447fe..9383a1b 100644
--- a/project/animates/thunder.animate
+++ b/project/animates/thunder.animate
@@ -1 +1 @@
-{"ratio":1,"se":"124-Thunder02.ogg","bitmaps":["","","","","","","","","",""],"frame_max":16,"frames":[[[2,0,-56,50,150,0,0]],[[2,0,-40,70,200,0,0]],[[2,0,-8,100,255,0,0]],[[3,0,-8,100,150,0,0]],[[3,0,-8,100,255,0,0]],[[2,0,-8,100,255,0,0]],[[2,0,-8,100,160,0,0]],[[2,0,-8,100,255,0,0]],[[3,0,-8,100,255,0,0]],[[3,0,-8,100,200,0,0]],[[0,0,-8,100,255,0,0]],[[0,0,-8,100,150,0,0]],[[1,0,-8,100,200,0,0]],[[1,0,-8,100,150,0,0]],[[0,0,-8,100,100,0,0]],[]]}
\ No newline at end of file
+{"ratio":1,"bitmaps":["","","","","","","","","","","",""],"frame_max":12,"frames":[[[0,0,-56,50,150,0,0]],[[1,0,-56,50,150,0,0]],[[2,0,-56,50,150,0,0]],[[3,0,-40,70,200,0,0]],[[4,0,-40,70,200,0,0]],[[5,0,-40,70,200,0,0]],[[6,0,-8,100,255,0,0]],[[7,0,-8,100,255,0,0]],[[8,0,-8,100,255,0,0]],[[9,0,-8,100,150,0,0]],[[10,0,-8,100,150,0,0]],[[11,0,-8,100,150,0,0]]],"se":{},"pitch":{}}
\ No newline at end of file
diff --git a/project/animates/yongchang.animate b/project/animates/yongchang.animate
index c4365f4..2173c5d 100644
--- a/project/animates/yongchang.animate
+++ b/project/animates/yongchang.animate
@@ -1 +1 @@
-{"ratio":2,"bitmaps":["","","","","","","","","",""],"frame_max":16,"frames":[[[2,0,-24,50,120]],[[3,0,-24,50,120]],[[2,0,-24,50,255]],[[3,0,-24,50,255],[5,-56,40,50,255],[5,0,64,30,255],[5,56,32,40,255]],[[4,0,-24,50,255],[5,56,8,40,255],[5,-32,32,30,255],[5,-56,8,50,255],[5,0,40,30,255]],[[3,0,-24,50,255],[5,-56,-16,50,255],[5,56,-16,40,255],[5,0,24,30,255],[5,-32,0,30,255]],[[4,0,-24,50,255],[5,-56,-32,50,255],[5,56,-32,40,255],[5,0,8,30,255],[5,-32,-16,30,255]],[[3,0,-24,50,255],[5,-56,-48,50,255],[5,56,-48,40,255],[5,0,8,30,255],[5,-32,-24,30,255]],[[2,0,-24,50,255],[5,-56,-64,50,150],[5,56,-56,40,150],[5,0,-16,30,150],[5,-32,-32,30,150]],[[3,0,-24,50,255],[5,-56,-80,50,150],[5,56,-72,40,150],[5,0,-32,30,150],[5,-32,-48,30,150]],[[4,0,-24,50,120],[5,-56,-96,50,80],[5,56,-96,40,100],[5,-32,-64,30,100],[5,0,-48,30,100]],[[4,0,-24,50,255],[5,-56,-112,50,80],[5,56,-112,40,100],[5,-32,-80,30,100],[5,0,-72,30,100]],[[3,0,-24,50,255]],[[2,0,-24,50,255]],[[2,0,-24,50,80]],[[2,0,-24,50,80]]]}
\ No newline at end of file
+{"ratio": 2.0, "bitmaps": ["", "", "", "", "", "", "", "", "", "", "", "", "", "", "", ""], "frame_max": 16, "frames": [[[2, 0, -24, 50, 120, 0, 0]], [[3, 0, -24, 50, 120, 0, 0]], [[2, 0, -24, 50, 255]], [[3, 0, -24, 50, 255], [5, -56, 40, 50, 255], [5, 0, 64, 30, 255], [5, 56, 32, 40, 255]], [[4, 0, -24, 50, 255], [5, 56, 8, 40, 255], [5, -32, 32, 30, 255], [5, -56, 8, 50, 255], [5, 0, 40, 30, 255]], [[3, 0, -24, 50, 255], [5, -56, -16, 50, 255], [5, 56, -16, 40, 255], [5, 0, 24, 30, 255], [5, -32, 0, 30, 255]], [[4, 0, -24, 50, 255, 0, 0], [5, -56, -32, 50, 255, 0, 0], [5, 56, -32, 40, 255, 0, 0], [5, 0, 8, 30, 255, 0, 0], [5, -32, -16, 30, 255, 0, 0]], [[3, 0, -24, 50, 255, 0, 0], [5, -56, -48, 50, 255, 0, 0], [5, 56, -48, 40, 255, 0, 0], [5, 0, 8, 30, 255, 0, 0], [5, -32, -24, 30, 255, 0, 0]], [[2, 0, -24, 50, 255, 0, 0], [5, -56, -64, 50, 150, 0, 0], [5, 56, -56, 40, 150, 0, 0], [5, 0, -16, 30, 150, 0, 0], [5, -32, -32, 30, 150, 0, 0]], [[3, 0, -24, 50, 255, 0, 0], [5, -56, -80, 50, 150, 0, 0], [5, 56, -72, 40, 150, 0, 0], [5, 0, -32, 30, 150, 0, 0], [5, -32, -48, 30, 150, 0, 0]], [[4, 0, -24, 50, 120, 0, 0], [5, -56, -96, 50, 80, 0, 0], [5, 56, -96, 40, 100, 0, 0], [5, -32, -64, 30, 100, 0, 0], [5, 0, -48, 30, 100, 0, 0]], [[4, 0, -24, 50, 255, 0, 0], [5, -56, -112, 50, 80, 0, 0], [5, 56, -112, 40, 100, 0, 0], [5, -32, -80, 30, 100, 0, 0], [5, 0, -72, 30, 100, 0, 0]], [[3, 0, -24, 50, 255, 0, 0]], [[2, 0, -24, 50, 255, 0, 0]], [[2, 0, -24, 50, 80, 0, 0]], [[2, 0, -24, 50, 80, 0, 0]]]}
\ No newline at end of file
diff --git a/project/animates/zone.animate b/project/animates/zone.animate
index 8c2b604..fa88a37 100644
--- a/project/animates/zone.animate
+++ b/project/animates/zone.animate
@@ -1 +1 @@
-{"ratio":2,"se":"zone.mp3","bitmaps":["","","","","","","","","",""],"frame_max":6,"frames":[[[0,0,32,30,100]],[[0,0,16,50,130]],[[0,0,0,80,150]],[[0,0,-8,90,180],[1,24,24,100,255]],[[0,0,-8,100,200],[1,24,8,100,255]],[[0,0,-8,100,255]]]}
\ No newline at end of file
+{"ratio": 2.0, "bitmaps": ["", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", ""], "frame_max": 18, "frames": [[[0, 0, 32, 30, 100]], [[0, 0, 32, 30, 100]], [[0, 0, 32, 30, 100]], [[0, 0, 16, 50, 130]], [[0, 0, 16, 50, 130]], [[0, 0, 16, 50, 130]], [[0, 0, 0, 80, 150, 0, 0]], [[0, 0, 0, 80, 150, 0, 0]], [[0, 0, 0, 80, 150, 0, 0]], [[0, 0, -8, 90, 180, 0, 0], [1, 24, 24, 100, 255, 0, 0]], [[0, 0, -8, 90, 180, 0, 0], [1, 24, 24, 100, 255, 0, 0]], [[0, 0, -8, 90, 180, 0, 0], [1, 24, 24, 100, 255, 0, 0]], [[0, 0, -8, 100, 200, 0, 0], [1, 24, 8, 100, 255, 0, 0]], [[0, 0, -8, 100, 200, 0, 0], [1, 24, 8, 100, 255, 0, 0]], [[0, 0, -8, 100, 200, 0, 0], [1, 24, 8, 100, 255, 0, 0]], [[0, 0, -8, 100, 255, 0, 0]], [[0, 0, -8, 100, 255, 0, 0]], [[0, 0, -8, 100, 255, 0, 0]]], "se": {"1": "zone.opus"}, "pitch": {"1": 100}}
\ No newline at end of file
diff --git a/project/bgms/Ascension.opus b/project/bgms/Ascension.opus
new file mode 100644
index 0000000..4ea0523
Binary files /dev/null and b/project/bgms/Ascension.opus differ
diff --git a/project/bgms/Blind_Alley.opus b/project/bgms/Blind_Alley.opus
index 92df164..0e37b2f 100644
Binary files a/project/bgms/Blind_Alley.opus and b/project/bgms/Blind_Alley.opus differ
diff --git a/project/bgms/Blood_Stain.opus b/project/bgms/Blood_Stain.opus
new file mode 100644
index 0000000..8d34051
Binary files /dev/null and b/project/bgms/Blood_Stain.opus differ
diff --git a/project/bgms/Halbmond.opus b/project/bgms/Halbmond.opus
new file mode 100644
index 0000000..cf8ae13
Binary files /dev/null and b/project/bgms/Halbmond.opus differ
diff --git a/project/bgms/ed.opus b/project/bgms/ed.opus
new file mode 100644
index 0000000..604d512
Binary files /dev/null and b/project/bgms/ed.opus differ
diff --git a/project/fonts/simhei.ttf b/project/bgms/theme.mp3
similarity index 54%
rename from project/fonts/simhei.ttf
rename to project/bgms/theme.mp3
index 5bd4687..3d4eeb3 100644
Binary files a/project/fonts/simhei.ttf and b/project/bgms/theme.mp3 differ
diff --git a/project/bgms/theme.opus b/project/bgms/theme.opus
deleted file mode 100644
index 5a178d6..0000000
Binary files a/project/bgms/theme.opus and /dev/null differ
diff --git a/project/data.js b/project/data.js
index 5b8e9ed..b3de946 100644
--- a/project/data.js
+++ b/project/data.js
@@ -2,6 +2,7 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 {
 	"main": {
 		"floorIds": [
+			"chushi",
 			"Caimhome",
 			"jiuguan",
 			"KTV",
@@ -11,11 +12,9 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"jiaotang",
 			"shengnvhome",
 			"shinvhome",
-			"jiedao",
+			"changguan2",
 			"street01",
 			"street02",
-			"guangchang",
-			"guangchang2",
 			"xiaoxiang01",
 			"xiaoxiang02",
 			"xiaoxiang3",
@@ -28,10 +27,25 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"yiqu7",
 			"yiqu8",
 			"yiqu9",
-			"yiqu10"
+			"yiqu10",
+			"changguan1",
+			"yushou",
+			"zhujuejia",
+			"guangchang",
+			"guangchang2",
+			"guangchang3",
+			"guangchang4",
+			"guangchang6",
+			"guangchang5",
+			"jiaocheng01",
+			"jiaocheng02",
+			"jiaocheng03"
 		],
 		"floorPartitions": [],
 		"images": [
+			"005-Attack03.webp",
+			"012-Heal01.webp",
+			"015-Fire01.webp",
 			"HPGaugeEnemy_A.webp",
 			"HPGaugeEnemy_B.webp",
 			"LOGO.webp",
@@ -432,19 +446,23 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"face_440110.webp",
 			"green.webp",
 			"hero.webp",
+			"jianji.webp",
 			"l.webp",
 			"lane1.webp",
 			"light.webp",
 			"lock.webp",
 			"lr.webp",
 			"maba.webp",
+			"miwu.webp",
 			"null.webp",
 			"other_0001.webp",
 			"other_0002.webp",
 			"other_0003.webp",
 			"other_0004.webp",
 			"r.webp",
+			"sound.webp",
 			"status.webp",
+			"statusBackground.webp",
 			"suiji.webp",
 			"tati_020101.webp",
 			"tati_020101a.webp",
@@ -776,6 +794,7 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"tati_120128.webp",
 			"tati_120131.webp",
 			"tati_120132.webp",
+			"tati_310101.webp",
 			"tati_340101.webp",
 			"tati_340101a.webp",
 			"tati_340101y.webp",
@@ -886,22 +905,29 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"8.webp",
 			"7.webp",
 			"10.webp",
-			"C5.webp"
+			"C5.webp",
+			"c6.png",
+			"2233.png"
 		],
 		"animates": [
 			"hand",
 			"jianji",
+			"jianji2",
 			"sword",
 			"thunder",
 			"yongchang",
 			"zone"
 		],
 		"bgms": [
+			"Ascension.opus",
 			"Asphodelus_Ceui.opus",
 			"Blind_Alley.opus",
+			"Blood_Stain.opus",
 			"Crawler.opus",
+			"Halbmond.opus",
+			"ed.opus",
 			"op.opus",
-			"theme.opus"
+			"theme.mp3"
 		],
 		"sounds": [
 			"aiy010000010.opus",
@@ -926,6 +952,143 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"aiy020000160.opus",
 			"aiy020000170.opus",
 			"aiy020000180.opus",
+			"aiy020000190.ogg",
+			"aiy020000200.ogg",
+			"aiy020000210.ogg",
+			"aiy020000220.ogg",
+			"aiy020000230.ogg",
+			"aiy020000240.ogg",
+			"aiy020000250.ogg",
+			"aiy020000260.ogg",
+			"aiy020000270.ogg",
+			"aiy020000280.ogg",
+			"aiy020000290.ogg",
+			"aiy020000300.ogg",
+			"aiy020000310.ogg",
+			"aiy020000320.ogg",
+			"aiy020000330.ogg",
+			"aiy020000340.ogg",
+			"aiy020000350.ogg",
+			"aiy020000360.ogg",
+			"aiy020000370.ogg",
+			"aiy030000010.ogg",
+			"aiy030000020.ogg",
+			"aiy030000030.ogg",
+			"aiy030000050.ogg",
+			"aiy030000060.ogg",
+			"aiy030000070.ogg",
+			"aiy030000080.ogg",
+			"aiy030000090.ogg",
+			"aiy030000100.ogg",
+			"aiy030000110.ogg",
+			"aiy030000120.ogg",
+			"aiy030000130.ogg",
+			"aiy050000010.ogg",
+			"aiy050000020.ogg",
+			"aiy050000030.ogg",
+			"aiy050000040.ogg",
+			"aiy050000050.ogg",
+			"aiy050000060.ogg",
+			"aiy050000070.ogg",
+			"aiy050000080.ogg",
+			"aiy050000090.ogg",
+			"aiy050000100.ogg",
+			"aiy050000110.ogg",
+			"aiy050000120.ogg",
+			"aiy120000020.ogg",
+			"aiy120000030.ogg",
+			"aiy120000040.ogg",
+			"aiy120000050.ogg",
+			"aiy120000060.ogg",
+			"aiy120000070.ogg",
+			"aiy120000080.ogg",
+			"aiy120000090.ogg",
+			"aiy120000100.ogg",
+			"aiy120000110.ogg",
+			"aiy120000120.ogg",
+			"aiy120000130.ogg",
+			"aiy120000140.ogg",
+			"aiy120000150.ogg",
+			"aiy120000160.ogg",
+			"aiy120000170.ogg",
+			"aiy120000180.ogg",
+			"aiy120000190.ogg",
+			"aiy120000200.ogg",
+			"aiy120000210.ogg",
+			"aiy120000220.ogg",
+			"aiy120000230.ogg",
+			"aiy120000240.ogg",
+			"aiy120000250.ogg",
+			"aiy120000260.ogg",
+			"aiy120000270.ogg",
+			"aiy120000280.ogg",
+			"aiy120000290.ogg",
+			"aiy120000300.ogg",
+			"aiy120000310.ogg",
+			"aiy120000320.ogg",
+			"aiy120000330.ogg",
+			"aiy120000340.ogg",
+			"aiy120000350.ogg",
+			"aiy120000360.ogg",
+			"aiy120000370.ogg",
+			"aiy120000380.ogg",
+			"aiy120000390.ogg",
+			"aiy120000400.ogg",
+			"aiy120000410.ogg",
+			"aiy120000420.ogg",
+			"aiy120000430.ogg",
+			"aiy120000440.ogg",
+			"aiy120000450.ogg",
+			"aiy120000460.ogg",
+			"aiy120000470.ogg",
+			"aiy120000480.ogg",
+			"aiy120000490.ogg",
+			"aiy120000500.ogg",
+			"aiy120000510.ogg",
+			"aiy120000520.ogg",
+			"aiy120000530.ogg",
+			"aiy120000540.ogg",
+			"aiy120000550.ogg",
+			"aiy120000560.ogg",
+			"aiy120000570.ogg",
+			"aiy120000580.ogg",
+			"aiy120000590.ogg",
+			"aiy120000600.ogg",
+			"aiy120000610.ogg",
+			"aiy120000620.ogg",
+			"aiy120000630.ogg",
+			"aiy120000640.ogg",
+			"aiy120000650.ogg",
+			"aiy120000670.ogg",
+			"aiy120000680.ogg",
+			"aiy120000690.ogg",
+			"aiy130000010.ogg",
+			"aiy130000020.ogg",
+			"aiy150000010.ogg",
+			"aiy150000020.ogg",
+			"aiy150000030.ogg",
+			"aiy150000040.ogg",
+			"aiy150000050.ogg",
+			"aiy150000060.ogg",
+			"aiy150000070.ogg",
+			"aiy150000080.ogg",
+			"aiy150000090.ogg",
+			"aiy150000100.ogg",
+			"aiy150000110.ogg",
+			"aiy150000120.ogg",
+			"aiy150000130.ogg",
+			"aiy150000140.ogg",
+			"aiy150000150.ogg",
+			"aiy150000160.ogg",
+			"aiy150000170.ogg",
+			"aiy150000180.ogg",
+			"aiy150000190.ogg",
+			"aiy150000200.ogg",
+			"aiy150000210.ogg",
+			"aiy150000220.ogg",
+			"aiy150000230.ogg",
+			"aiy150000240.ogg",
+			"aiy150000250.ogg",
 			"aiy310000010.opus",
 			"aiy310000020.opus",
 			"aiy310000030.opus",
@@ -955,6 +1118,170 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"aiy310000280.opus",
 			"aiy310000290.opus",
 			"aiy310000300.opus",
+			"aiy310000310.ogg",
+			"aiy310000320.ogg",
+			"aiy310000330.ogg",
+			"aiy310000340.ogg",
+			"aiy310000350.ogg",
+			"aiy310000360.ogg",
+			"aiy310000370.ogg",
+			"aiy310000380.ogg",
+			"aiy310000390.ogg",
+			"aiy310000400.ogg",
+			"aiy310000410.ogg",
+			"aiy310000420.ogg",
+			"aiy310000430.ogg",
+			"aiy310000440.ogg",
+			"aiy310000450.ogg",
+			"aiy310000460.ogg",
+			"aiy310000470.ogg",
+			"aiy310000480.ogg",
+			"aiy310000490.ogg",
+			"aiy310000510.ogg",
+			"aiy310000520.ogg",
+			"aiy310000530.ogg",
+			"aiy310000540.ogg",
+			"aiy310000550.ogg",
+			"aiy310000560.ogg",
+			"aiy310000570.ogg",
+			"aiy310000580.ogg",
+			"aiy310000590.ogg",
+			"aiy310000600.ogg",
+			"aiy310000610.ogg",
+			"aiy310000620.ogg",
+			"aiy310000630.ogg",
+			"aiy310000640.ogg",
+			"aiy310000650.ogg",
+			"aiy310000660.ogg",
+			"aiy310000670.ogg",
+			"aiy310000680.ogg",
+			"aiy310000690.ogg",
+			"aiy310000700.ogg",
+			"aiy310000710.ogg",
+			"aiy310000720.ogg",
+			"aiy310000730.ogg",
+			"aiy310000740.ogg",
+			"aiy310000750.ogg",
+			"aiy310000760.ogg",
+			"aiy310000770.ogg",
+			"aiy310000780.ogg",
+			"aiy310000790.ogg",
+			"aiy310000800.ogg",
+			"aiy310000810.ogg",
+			"aiy310000820.ogg",
+			"aiy310000830.ogg",
+			"aiy310000840.ogg",
+			"aiy310000850.ogg",
+			"aiy310000860.ogg",
+			"aiy310000870.ogg",
+			"aiy310000880.ogg",
+			"aiy310000890.ogg",
+			"aiy310000900.ogg",
+			"aiy310000910.ogg",
+			"aiy310000920.ogg",
+			"aiy310000930.ogg",
+			"aiy310000940.ogg",
+			"aiy310000950.ogg",
+			"aiy310000960.ogg",
+			"aiy310000970.ogg",
+			"aiy310000980.ogg",
+			"aiy310000990.ogg",
+			"aiy310001000.ogg",
+			"aiy310001010.ogg",
+			"aiy310001027.ogg",
+			"aiy310001030.ogg",
+			"aiy310001040.ogg",
+			"aiy310001050.ogg",
+			"aiy310001060.ogg",
+			"aiy310001070.ogg",
+			"aiy310001080.ogg",
+			"aiy310001090.ogg",
+			"aiy310001100.ogg",
+			"aiy310001110.ogg",
+			"aiy310001120.ogg",
+			"aiy310001130.ogg",
+			"aiy310001140.ogg",
+			"aiy310001150.ogg",
+			"aiy310001160.ogg",
+			"aiy310001170.ogg",
+			"aiy310001180.ogg",
+			"aiy310001190.ogg",
+			"aiy310001200.ogg",
+			"aiy310001210.ogg",
+			"aiy310001220.ogg",
+			"aiy310001230.ogg",
+			"aiy310001240.ogg",
+			"aiy310001250.ogg",
+			"aiy310001260.ogg",
+			"aiy310001270.ogg",
+			"aiy310001280.ogg",
+			"aiy310001290.ogg",
+			"aiy310001300.ogg",
+			"aiy310001310.ogg",
+			"aiy310001320.ogg",
+			"aiy310001330.ogg",
+			"aiy310001340.ogg",
+			"aiy310001350.ogg",
+			"aiy310001360.ogg",
+			"aiy310001370.ogg",
+			"aiy310001380.ogg",
+			"aiy310001390.ogg",
+			"aiy320000010.ogg",
+			"aiy320000020.ogg",
+			"aiy320000030.ogg",
+			"aiy320000040.ogg",
+			"aiy320000050.ogg",
+			"aiy320000060.ogg",
+			"aiy320000070.ogg",
+			"aiy320000080.ogg",
+			"aiy320000090.ogg",
+			"aiy320000100.ogg",
+			"aiy320000110.ogg",
+			"aiy320000120.ogg",
+			"aiy320000135.ogg",
+			"aiy320000140.ogg",
+			"aiy340000010.ogg",
+			"aiy340000020.ogg",
+			"aiy340000030.ogg",
+			"aiy340000040.ogg",
+			"aiy340000050.ogg",
+			"aiy340000060.ogg",
+			"aiy340000070.ogg",
+			"aiy340000080.ogg",
+			"aiy340000090.ogg",
+			"aiy340000100.ogg",
+			"aiy340000110.ogg",
+			"aiy340000120.ogg",
+			"aiy340000140.ogg",
+			"aiy340000150.ogg",
+			"aiy340000160.ogg",
+			"aiy340000170.ogg",
+			"aiy340000180.ogg",
+			"aiy340000190.ogg",
+			"aiy340000200.ogg",
+			"aiy340000210.ogg",
+			"aiy340000220.ogg",
+			"aiy340000230.ogg",
+			"aiy340000240.ogg",
+			"aiy340000250.ogg",
+			"aiy340000260.ogg",
+			"aiy340000270.ogg",
+			"aiy340000280.ogg",
+			"aiy340000290.ogg",
+			"aiy340000300.ogg",
+			"aiy340000310.ogg",
+			"aiy340000320.ogg",
+			"aiy340000330.ogg",
+			"aiy340000340.ogg",
+			"aiy340000350.ogg",
+			"aiy340000360.ogg",
+			"aiy340000370.ogg",
+			"aiy340000380.ogg",
+			"aiy340000390.ogg",
+			"aiy340000400.ogg",
+			"aiy340000410.ogg",
+			"aiy340000420.ogg",
+			"aiy340000430.ogg",
 			"aiy350000010.opus",
 			"aiy350000020.opus",
 			"aiy350000030.opus",
@@ -978,6 +1305,52 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"aiy350000210.opus",
 			"aiy350000220.opus",
 			"aiy350000230.opus",
+			"aiy350000240.ogg",
+			"aiy350000250.ogg",
+			"aiy350000260.ogg",
+			"aiy430000010.ogg",
+			"aiy430000020.ogg",
+			"aiy430000030.ogg",
+			"aiy430000040.ogg",
+			"aiy430000050.ogg",
+			"aiy430000060.ogg",
+			"aiy430000070.ogg",
+			"aiy430000080.ogg",
+			"aiy430000090.ogg",
+			"aiy430000100.ogg",
+			"aiy430000110.ogg",
+			"aiy430000120.ogg",
+			"aiy430000130.ogg",
+			"aiy430000140.ogg",
+			"aiy430000150.ogg",
+			"aiy430000160.ogg",
+			"aiy430000170.ogg",
+			"aiy430000180.ogg",
+			"aiy430000190.ogg",
+			"aiy430000200.ogg",
+			"aiy430000210.ogg",
+			"aiy430000220.ogg",
+			"aiy430000230.ogg",
+			"aiy430000240.ogg",
+			"aiy430000250.ogg",
+			"aiy430000260.ogg",
+			"aiy430000270.ogg",
+			"aiy430000280.ogg",
+			"aiy430000290.ogg",
+			"aiy440000010.ogg",
+			"aiy440000020.ogg",
+			"aiy440000030.ogg",
+			"aiy440000040.ogg",
+			"aiy440000050.ogg",
+			"aiy440000060.ogg",
+			"aiy440000070.ogg",
+			"aiy440000080.ogg",
+			"aiy440000090.ogg",
+			"aiy440000100.ogg",
+			"aiy440000110.ogg",
+			"aiy440000120.ogg",
+			"aiy510000010.ogg",
+			"aiy510000020.ogg",
 			"aiy710000010.opus",
 			"aiy710000020.opus",
 			"aiy710000030.opus",
@@ -991,9 +1364,68 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"aiy710000110.opus",
 			"aiy710000120.opus",
 			"aiy710000130.opus",
+			"aiy720000010.ogg",
+			"aiy730000010.ogg",
+			"aiy740000010.ogg",
+			"aiy750000010.ogg",
+			"aiy750000020.ogg",
+			"aiy750000030.ogg",
+			"aiy750000040.ogg",
+			"aiy750000050.ogg",
+			"aiy750000060.ogg",
+			"aiy750000070.ogg",
+			"aiy750000080.ogg",
+			"aiy750000090.ogg",
+			"aiy750000100.ogg",
+			"aiy750000110.ogg",
+			"aiy750000120.ogg",
+			"aiy750000130.ogg",
+			"aiy750000140.ogg",
+			"aiy750000150.ogg",
+			"aiy750000160.ogg",
+			"aiy750000170.ogg",
+			"aiy750000180.ogg",
+			"aiy750000190.ogg",
+			"aiy750000200.ogg",
+			"aiy750000210.ogg",
+			"aiy750000220.ogg",
+			"aiy750000230.ogg",
+			"aiy750000240.ogg",
+			"aiy750000250.ogg",
+			"aiy750000260.ogg",
+			"aiy750000270.ogg",
+			"aiy750000280.ogg",
+			"aiy750000290.ogg",
+			"aiy750000300.ogg",
+			"aiy750000310.ogg",
+			"aiy750000320.ogg",
+			"aiy750000330.ogg",
+			"aiy750000340.ogg",
+			"aiy800000010.ogg",
+			"aiy800000020.ogg",
+			"aiy800000030.ogg",
+			"aiy800000040.ogg",
+			"aiy800000050.ogg",
+			"aiy800000060.ogg",
+			"aiy800000070.ogg",
+			"aiy800000080.ogg",
+			"aiy800000090.ogg",
+			"aiy800000100.ogg",
+			"aiy810000010.ogg",
+			"aiy810000020.ogg",
+			"aiy810000030.ogg",
+			"aiy810000040.ogg",
+			"aiy810000050.ogg",
+			"aiy810000060.ogg",
+			"aiy810000070.ogg",
 			"aiy820000010.opus",
 			"aiy820000020.opus",
-			"attack.opus",
+			"aiy860000010.ogg",
+			"aiy860000020.ogg",
+			"aiy860000030.ogg",
+			"aiy860000040.ogg",
+			"aiy860000050.ogg",
+			"aiy860000060.ogg",
 			"attack.opus",
 			"bomb.opus",
 			"cancel.opus",
@@ -1019,9 +1451,9 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 		],
 		"fonts": [
 			"HATTEN",
+			"Verdana",
 			"number",
-			"pala",
-			"simhei"
+			"pala"
 		],
 		"nameMap": {
 			"确定": "confirm.opus",
@@ -1057,7 +1489,7 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			"饰品",
 			"饰品"
 		],
-		"startBgm": "theme.opus",
+		"startBgm": "theme.mp3",
 		"styles": {
 			"startBackground": "project/images/background.webp",
 			"startVerticalBackground": "project/images/backgroundvertical.webp",
@@ -1091,22 +1523,22 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 	},
 	"firstData": {
 		"title": "秽翼的尤斯蒂娅",
-		"name": "Eustia",
+		"name": "Eustia_text",
 		"version": "鸽子窝造塔小队",
-		"floorId": "street01",
+		"floorId": "chushi",
 		"hero": {
 			"image": "hero.webp",
 			"animate": false,
-			"name": "Caim",
+			"name": "凯伊姆",
 			"lv": 1,
 			"hpmax": 9999,
 			"hp": 1000,
 			"manamax": -1,
 			"mana": 0,
-			"atk": 100,
-			"def": 100,
-			"mdef": 0,
-			"speed": 0,
+			"atk": 26,
+			"def": 16,
+			"mdef": 100,
+			"speed": 16,
 			"money": 0,
 			"exp": 0,
 			"equipment": [],
@@ -1117,15 +1549,21 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			},
 			"loc": {
 				"direction": "up",
-				"x": 5,
-				"y": 11
+				"x": 6,
+				"y": 6
 			},
 			"flags": {
 				"itemDetail": true,
-				"useBetweenLight": true
+				"useBetweenLight": true,
+				"__mdef_buff__": 0,
+				"popmove": true
 			},
 			"followers": [],
-			"steps": 0
+			"steps": 0,
+			"matk": 0,
+			"spell": 0,
+			"spelldef": 0,
+			"mhp": 100
 		},
 		"startCanvas": [
 			{
@@ -1441,7 +1879,7 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 			},
 			{
 				"type": "function",
-				"function": "function(){\ncore.control.checkBgm()\n}"
+				"function": "function(){\ncore.checkBgm()\n}"
 			},
 			{
 				"type": "if",
@@ -1481,14 +1919,30 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 		],
 		"startText": [
 			{
-				"type": "setValue",
-				"name": "item:book",
-				"value": "1"
+				"type": "setText"
 			},
 			{
-				"type": "setValue",
-				"name": "item:fly",
-				"value": "1"
+				"type": "setHeroOpacity",
+				"opacity": 0
+			},
+			{
+				"type": "setCurtain",
+				"color": [
+					0,
+					0,
+					0,
+					1
+				],
+				"time": 500,
+				"keep": true
+			},
+			{
+				"type": "function",
+				"function": "function(){\ncore.getItem('book', 1);\ncore.getItem('fly', 1);\ncore.getItem('postman', 1);\ncore.getItem('I369', 1);\n}"
+			},
+			{
+				"type": "insert",
+				"name": "强制横屏"
 			}
 		],
 		"shops": [
@@ -1657,13 +2111,13 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 		"lavaDamage": 100,
 		"poisonDamage": 10,
 		"weakValue": 20,
-		"redGem": 3,
-		"blueGem": 3,
+		"redGem": 2,
+		"blueGem": 2,
 		"greenGem": 5,
 		"redPotion": 100,
 		"bluePotion": 250,
 		"yellowPotion": 500,
-		"greenPotion": 800,
+		"greenPotion": 900,
 		"breakArmor": 0.9,
 		"counterAttack": 0.1,
 		"purify": 3,
@@ -1671,7 +2125,8 @@ var data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d =
 		"animateSpeed": 300,
 		"moveSpeed": 100,
 		"statusCanvasRowsOnMobile": 3,
-		"floorChangeTime": 500
+		"floorChangeTime": 0,
+		"yellowGem": 2
 	},
 	"flags": {
 		"statusBarItems": [
diff --git a/project/enemys.js b/project/enemys.js
index 931f40a..5e8a014 100644
--- a/project/enemys.js
+++ b/project/enemys.js
@@ -1,84 +1,88 @@
 var enemys_fcae963b_31c9_42b4_b48c_bb48d09f3f80 = 
 {
-	"greenSlime": {"name":"绿头怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"redSlime": {"name":"红头怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"value":10},
-	"blackSlime": {"name":"青头怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"slimelord": {"name":"怪王","hp":100,"atk":120,"def":0,"money":10,"exp":0,"point":0,"special":[1,9]},
-	"bat": {"name":"小蝙蝠","hp":100,"atk":120,"def":0,"money":2,"exp":0,"point":0,"special":[1]},
-	"bigBat": {"name":"大蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"redBat": {"name":"红蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[]},
-	"vampire": {"name":"冥灵魔王","hp":888,"atk":888,"def":888,"money":888,"exp":888,"point":0,"special":[6],"n":8},
-	"skeleton": {"name":"骷髅人","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"skeletonCaptain": {"name":"骷髅队长","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"zombie": {"name":"兽人","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"magic":false},
-	"zombieKnight": {"name":"兽人武士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"rock": {"name":"石头人","hp":50,"atk":50,"def":0,"money":3,"exp":0,"point":0,"special":3},
-	"bluePriest": {"name":"初级法师","hp":100,"atk":120,"def":0,"money":3,"exp":0,"point":1,"special":[9]},
-	"redPriest": {"name":"高级法师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"brownWizard": {"name":"初级巫师","hp":100,"atk":120,"def":0,"money":16,"exp":0,"point":0,"special":15,"value":100,"range":2},
-	"redWizard": {"name":"高级巫师","hp":1000,"atk":1200,"def":0,"money":160,"exp":0,"point":0,"special":15,"value":200,"zoneSquare":true},
-	"swordsman": {"name":"双手剑士","hp":100,"atk":120,"def":0,"money":6,"exp":3,"point":0,"special":4},
-	"soldier": {"name":"冥战士","hp":120,"atk":50,"def":8,"money":10,"exp":7,"point":0,"special":0},
-	"yellowKnight": {"name":"金骑士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"redKnight": {"name":"红骑士","hp":500,"atk":200,"def":50,"money":0,"exp":0,"point":0,"special":[7]},
-	"darkKnight": {"name":"黑骑士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"blueKnight": {"name":"蓝骑士","hp":100,"atk":120,"def":0,"money":9,"exp":0,"point":0,"special":8},
-	"goldSlime": {"name":"黄头怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"poisonSkeleton": {"name":"紫骷髅","hp":50,"atk":60,"def":70,"money":80,"exp":0,"point":0,"special":13},
-	"poisonBat": {"name":"紫蝙蝠","hp":100,"atk":120,"def":0,"money":14,"exp":0,"point":0,"special":13},
-	"skeletonPriest": {"name":"骷髅法师","hp":100,"atk":100,"def":0,"money":0,"exp":0,"point":0,"special":18,"value":20},
-	"skeletonKing": {"name":"骷髅王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"evilHero": {"name":"迷失勇者","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"demonPriest": {"name":"魔神法师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"goldHornSlime": {"name":"金角怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"silverSlime": {"name":"银头怪","hp":100,"atk":120,"def":0,"money":15,"exp":0,"point":0,"special":14},
+	"greenSlime": {"name":"绿头怪","hp":64,"atk":32,"def":2,"money":0,"exp":1,"point":0,"special":[],"mdef":0,"spell":0,"speed":17},
+	"redSlime": {"name":"红头怪","hp":84,"atk":27,"def":6,"money":0,"exp":1,"point":0,"special":[],"value":null,"mdef":0,"spell":0,"speed":19},
+	"blackSlime": {"name":"青头怪","hp":222,"atk":46,"def":10,"money":0,"exp":0,"point":0,"special":[],"mdef":0,"spell":0,"speed":15,"damage":null},
+	"slimelord": {"name":"怪王","hp":790,"atk":48,"def":8,"money":0,"exp":0,"point":0,"special":[],"mdef":0,"spell":0,"speed":20},
+	"bat": {"name":"小蝙蝠","hp":46,"atk":49,"def":0,"money":0,"exp":1,"point":0,"special":[1],"mdef":0,"speed":29,"spell":0},
+	"bigBat": {"name":"大蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"redBat": {"name":"红蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"mdef":0,"spell":0,"speed":1},
+	"vampire": {"name":"冥灵魔王","hp":888,"atk":888,"def":888,"money":888,"exp":888,"point":0,"special":[6],"n":8,"mdef":0,"spell":0,"speed":1},
+	"skeleton": {"name":"骷髅人","hp":77,"atk":77,"def":7,"money":0,"exp":2,"point":0,"special":0,"mdef":0,"spell":0,"speed":17},
+	"skeletonCaptain": {"name":"骷髅队长","hp":226,"atk":66,"def":1,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":18},
+	"zombie": {"name":"兽人","hp":444,"atk":43,"def":9,"money":0,"exp":3,"point":0,"special":[],"mdef":0,"spell":0,"speed":27},
+	"zombieKnight": {"name":"兽人武士","hp":10000,"atk":100,"def":100,"money":0,"exp":0,"point":0,"special":[22],"speed":10,"mdef":0,"damage2":20,"spell":0},
+	"rock": {"name":"石头人","hp":50,"atk":50,"def":0,"money":3,"exp":0,"point":0,"special":3,"damage2":15,"mdef":0,"spell":0,"speed":1},
+	"bluePriest": {"name":"初级法师","hp":108,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"mdef":0,"spell":270,"speed":12},
+	"redPriest": {"name":"高级法师","hp":162,"atk":0,"def":1,"money":0,"exp":0,"point":0,"special":[1],"mdef":0,"spell":308,"speed":14},
+	"brownWizard": {"name":"初级巫师","hp":162,"atk":120,"def":1,"money":16,"exp":0,"point":0,"special":[],"value":null,"range":2,"mdef":0,"spell":66,"speed":14},
+	"redWizard": {"name":"高级巫师","hp":1000,"atk":1200,"def":0,"money":160,"exp":0,"point":0,"special":15,"value":200,"zoneSquare":true,"mdef":0,"spell":0,"speed":1},
+	"swordsman": {"name":"双手剑士","hp":100,"atk":120,"def":0,"money":6,"exp":3,"point":0,"special":4,"mdef":0,"spell":0,"speed":1},
+	"soldier": {"name":"冥战士","hp":120,"atk":50,"def":8,"money":10,"exp":7,"point":0,"special":0,"mdef":0,"spell":0,"speed":8},
+	"yellowKnight": {"name":"金骑士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"redKnight": {"name":"红骑士","hp":500,"atk":200,"def":50,"money":0,"exp":0,"point":0,"special":[7],"mdef":0,"spell":0,"speed":1},
+	"darkKnight": {"name":"黑骑士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"blueKnight": {"name":"蓝骑士","hp":100,"atk":120,"def":0,"money":9,"exp":0,"point":0,"special":8,"mdef":0,"spell":0,"speed":1},
+	"goldSlime": {"name":"黄头怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"poisonSkeleton": {"name":"紫骷髅","hp":50,"atk":60,"def":70,"money":80,"exp":0,"point":0,"special":13,"mdef":0,"spell":0,"speed":1},
+	"poisonBat": {"name":"紫蝙蝠","hp":100,"atk":120,"def":0,"money":14,"exp":0,"point":0,"special":13,"mdef":0,"spell":0,"speed":1},
+	"skeletonPriest": {"name":"骷髅法师","hp":100,"atk":100,"def":0,"money":0,"exp":0,"point":0,"special":18,"value":20,"mdef":0,"spell":0,"speed":1},
+	"skeletonKing": {"name":"骷髅王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"evilHero": {"name":"迷失勇者","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"demonPriest": {"name":"魔神法师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"goldHornSlime": {"name":"小恶魔","hp":126,"atk":48,"def":12,"money":0,"exp":2,"point":0,"special":0,"mdef":0,"spell":0,"speed":16},
+	"silverSlime": {"name":"银头怪","hp":100,"atk":120,"def":0,"money":15,"exp":0,"point":0,"special":14,"mdef":0,"spell":0,"speed":1},
 	"whiteHornSlime": {"name":"尖角怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"redSwordsman": {"name":"剑王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[]},
-	"poisonZombie": {"name":"绿兽人","hp":100,"atk":120,"def":0,"money":13,"exp":0,"point":0,"special":[12]},
-	"octopus": {"name":"血影","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"bigImage":null},
-	"princessEnemy": {"name":"假公主","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"angel": {"name":"天使","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"elemental": {"name":"元素生物","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"steelGuard": {"name":"铁守卫","hp":50,"atk":50,"def":50,"money":0,"exp":0,"point":0,"special":[18],"value":20},
-	"evilBat": {"name":"邪恶蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"silverSlimelord": {"name":"银怪王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"goldSlimelord": {"name":"金怪王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"skeletonWarrior": {"name":"骷髅士兵","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"whiteSlimeman": {"name":"水银战士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"slimeman": {"name":"影子战士","hp":100,"atk":0,"def":0,"money":11,"exp":0,"point":0,"special":[9],"atkValue":2,"defValue":3},
-	"yellowGateKeeper": {"name":"初级卫兵","hp":80,"atk":25,"def":5,"money":3,"exp":1,"point":0,"special":0},
-	"blueGateKeeper": {"name":"中级卫兵","hp":80,"atk":25,"def":5,"money":3,"exp":1,"point":0,"special":0},
-	"redGateKeeper": {"name":"高级卫兵","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"magicMaster": {"name":"黑暗大法师","hp":100,"atk":120,"def":0,"money":12,"exp":0,"point":0,"special":11,"value":0.3333333333333333,"add":true,"notBomb":true},
-	"devilWarrior": {"name":"魔神武士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"fairyEnemy": {"name":"仙子","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"dragon": {"name":"魔龙","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"bigImage":null},
-	"skeletonKnight": {"name":"骷髅武士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"skeletonPresbyter": {"name":"骷髅巫师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"ironRock": {"name":"铁面人","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"grayRock": {"name":"灰色石头人","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"yellowPriest": {"name":"中级法师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"evilPrincess": {"name":"痛苦魔女","hp":1000,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[10]},
+	"redSwordsman": {"name":"剑王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"mdef":0,"spell":0,"speed":1},
+	"poisonZombie": {"name":"绿兽人","hp":100,"atk":120,"def":0,"money":13,"exp":0,"point":0,"special":[12],"mdef":0,"spell":0,"speed":1},
+	"octopus": {"name":"血影","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"bigImage":null,"mdef":0,"spell":0,"speed":1},
+	"princessEnemy": {"name":"假公主","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"angel": {"name":"天使","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"elemental": {"name":"元素生物","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"steelGuard": {"name":"铁守卫","hp":50,"atk":50,"def":50,"money":0,"exp":0,"point":0,"special":[18],"value":20,"mdef":0,"spell":0,"speed":1},
+	"evilBat": {"name":"邪恶蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"silverSlimelord": {"name":"银怪王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"goldSlimelord": {"name":"金怪王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"skeletonWarrior": {"name":"骷髅士兵","hp":226,"atk":66,"def":1,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":18,"displayIdInBook":null},
+	"whiteSlimeman": {"name":"水银战士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"slimeman": {"name":"影子战士","hp":243,"atk":28,"def":5,"money":0,"exp":0,"point":0,"special":[],"atkValue":2,"defValue":3,"mdef":0,"spell":0,"speed":69},
+	"yellowGateKeeper": {"name":"初级卫兵","hp":80,"atk":25,"def":5,"money":3,"exp":1,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"blueGateKeeper": {"name":"中级卫兵","hp":80,"atk":25,"def":5,"money":3,"exp":1,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"redGateKeeper": {"name":"高级卫兵","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"magicMaster": {"name":"黑暗大法师","hp":100,"atk":120,"def":0,"money":12,"exp":0,"point":0,"special":11,"value":0.3333333333333333,"add":true,"notBomb":true,"mdef":0,"spell":0,"speed":1},
+	"devilWarrior": {"name":"魔神武士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"fairyEnemy": {"name":"仙子","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"dragon": {"name":"魔龙","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"bigImage":null,"mdef":0,"spell":0,"speed":1},
+	"skeletonKnight": {"name":"骷髅武士","hp":330,"atk":113,"def":2,"money":0,"exp":4,"point":0,"special":0,"mdef":0,"spell":0,"speed":20},
+	"skeletonPresbyter": {"name":"骷髅巫师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"ironRock": {"name":"铁面人","hp":111,"atk":97,"def":37,"money":0,"exp":0,"point":0,"special":[3],"mdef":0,"spell":0,"speed":14},
+	"grayRock": {"name":"灰色石头人","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"yellowPriest": {"name":"中级法师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"evilPrincess": {"name":"痛苦魔女","hp":1000,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[10],"mdef":0,"spell":0,"speed":1},
 	"blademaster": {"name":"剑圣","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"evilFairy": {"name":"黑暗仙子","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"greenKnight": {"name":"强盾骑士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"bowman": {"name":"初级弓兵","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"watcherSlime": {"name":"邪眼怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"devilKnight": {"name":"恶灵骑士","hp":150,"atk":100,"def":50,"money":0,"exp":0,"point":0,"special":[1,5,7,8]},
-	"grayPriest": {"name":"混沌法师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"greenGateKeeper": {"name":"卫兵队长","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"ghostSoldier": {"name":"冥队长","hp":200,"atk":100,"def":50,"money":0,"exp":0,"point":0,"special":8},
-	"frostBat": {"name":"寒蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"blackKing": {"name":"黑衣魔王","hp":1000,"atk":500,"def":0,"money":1000,"exp":1000,"point":0,"special":0,"notBomb":true},
-	"yellowKing": {"name":"黄衣魔王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"greenKing": {"name":"青衣武士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"redKing": {"name":"红衣魔王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"blueKing": {"name":"白衣武士","hp":100,"atk":120,"def":0,"money":17,"exp":0,"point":0,"special":[16]},
-	"keiskeiFairy": {"name":"铃兰花妖","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"tulipFairy": {"name":"郁金香花妖","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"purpleBowman": {"name":"高级弓兵","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0},
-	"bearDown": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"},"bigImage":"bear.png"},
-	"bearLeft": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"}},
-	"bearRight": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"}},
-	"bearUp": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"}}
+	"evilFairy": {"name":"黑暗仙子","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"greenKnight": {"name":"强盾骑士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"bowman": {"name":"初级弓兵","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"watcherSlime": {"name":"邪眼怪","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"devilKnight": {"name":"恶灵骑士","hp":150,"atk":100,"def":50,"money":0,"exp":0,"point":0,"special":[1,5,7,8],"mdef":0,"spell":0,"speed":1},
+	"grayPriest": {"name":"混沌法师","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"greenGateKeeper": {"name":"卫兵队长","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"ghostSoldier": {"name":"冥队长","hp":200,"atk":100,"def":50,"money":0,"exp":0,"point":0,"special":8,"mdef":0,"spell":0,"speed":1},
+	"frostBat": {"name":"寒蝙蝠","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"blackKing": {"name":"黑衣魔王","hp":1000,"atk":500,"def":0,"money":1000,"exp":1000,"point":0,"special":0,"notBomb":true,"mdef":0,"spell":0,"speed":1},
+	"yellowKing": {"name":"黄衣魔王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"greenKing": {"name":"青衣武士","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"redKing": {"name":"红衣魔王","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"blueKing": {"name":"白衣武士","hp":100,"atk":120,"def":0,"money":17,"exp":0,"point":0,"special":[16],"mdef":0,"spell":0,"speed":1},
+	"keiskeiFairy": {"name":"铃兰花妖","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"tulipFairy": {"name":"郁金香花妖","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"purpleBowman": {"name":"高级弓兵","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":1},
+	"bearDown": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"},"bigImage":"bear.png","mdef":0,"spell":0,"speed":1},
+	"bearLeft": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"},"mdef":0,"spell":0,"speed":1},
+	"bearRight": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"},"mdef":0,"spell":0,"speed":1},
+	"bearUp": {"name":"熊出没","hp":0,"atk":0,"def":0,"money":0,"exp":0,"point":0,"special":[],"faceIds":{"down":"bearDown","left":"bearLeft","right":"bearRight","up":"bearUp"},"mdef":0,"spell":0,"speed":1},
+	"E421": {"name":"蓝头怪","hp":104,"atk":33,"def":8,"money":0,"exp":1,"point":0,"special":[],"value":null,"mdef":0,"spell":0,"speed":19,"displayIdInBook":null},
+	"E422": {"name":"新敌人","hp":0,"atk":0,"def":0,"mdef":0,"speed":1,"money":0,"exp":0,"point":0,"special":[]},
+	"E425": {"name":"黄骷髅","hp":226,"atk":66,"def":1,"money":0,"exp":0,"point":0,"special":0,"mdef":0,"spell":0,"speed":18,"displayIdInBook":null},
+	"E426": {"name":"小偷","hp":377,"atk":53,"def":17,"mdef":0,"speed":34,"money":50,"exp":10,"point":0,"special":[],"spell":0}
 }
\ No newline at end of file
diff --git a/project/events.js b/project/events.js
index 1e860d2..634691d 100644
--- a/project/events.js
+++ b/project/events.js
@@ -154,1051 +154,964 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 				"time": 50
 			},
 			{
-				"type": "changebg",
-				"img1": "",
-				"memory1": false,
-				"img2": "",
-				"memory2": false,
-				"time": 30,
-				"style": "引入"
-			},
-			{
-				"type": "over",
-				"text": "人生不如意事,十常居七八。",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "世间的万物皆被装缀着不公的色彩。",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "没错,就像——",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "十几年前的那一天,",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "在都市的角落,同许多的生命共同消逝一般。",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "没错,就像——",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "十几年前的那天起以来,",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "无数的生命被那无法逃脱的污泥囚禁一样。",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "人生不如意事,十常居七八。",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "世间的万物皆被装缀着不公的色彩。",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "没错,就像——",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "over",
-				"text": "如今,这一天。",
-				"image": "",
-				"memory": false,
-				"time": 50,
-				"hidetime": 30,
-				"sound": "",
-				"textColor": [
-					255,
-					255,
-					255,
-					1
-				],
-				"boldColor": [
-					0,
-					0,
-					0,
-					1
-				],
-				"font": "bold 48px Verdana"
-			},
-			{
-				"type": "changebg",
-				"img1": "",
-				"memory1": false,
-				"img2": "other_0001.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0001.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "这些天,街道不曾下雨。",
-				"bodyList": [
+				"type": "confirm",
+				"text": "跳过傻逼不能点点点快进的旁白",
+				"yes": [],
+				"no": [
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
+						"type": "changebg",
+						"img1": "",
+						"memory1": false,
+						"img2": "bg_0000.png",
+						"memory2": false,
+						"time": 30,
+						"style": "引入"
+					},
+					{
+						"type": "over",
+						"text": "人生不如意事,十常居七八。",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "世间的万物皆被装缀着不公的色彩。",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "没错,就像——",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "十几年前的那一天,",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "在都市的角落,同许多的生命共同消逝一般。",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "没错,就像——",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "十几年前的那天起以来,",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "无数的生命被那无法逃脱的污泥囚禁一样。",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "人生不如意事,十常居七八。",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "世间的万物皆被装缀着不公的色彩。",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "没错,就像——",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "over",
+						"text": "如今,这一天。",
+						"image": "",
+						"memory": false,
+						"time": 50,
+						"hidetime": 30,
+						"sound": "",
+						"textColor": [
+							255,
+							255,
+							255,
+							1
+						],
+						"boldColor": [
+							0,
+							0,
+							0,
+							1
+						],
+						"font": "bold 48px Verdana"
+					},
+					{
+						"type": "changebg",
+						"img1": "bg_0000.png",
+						"memory1": false,
+						"img2": "",
+						"memory2": false,
+						"time": 30,
+						"style": "引出"
 					}
 				]
 			},
 			{
-				"type": "cgtext",
-				"bg": "other_0001.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "所以,那浸湿地面的,定是那些女孩们流落的鲜血无疑。",
-				"bodyList": [
+				"type": "confirm",
+				"text": "我他妈要全跳了",
+				"yes": [],
+				"no": [
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0001.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "我蹲在充斥着铁锈味般恶臭的小巷中,悠闲地如是想着。",
-				"bodyList": [
+						"type": "changebg",
+						"img1": "",
+						"memory1": false,
+						"img2": "other_0001.webp",
+						"memory2": false,
+						"time": 30,
+						"style": "引入"
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "other_0001.webp",
-				"memory1": false,
-				"img2": "bg_3563.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "扑哧。",
-				"bodyList": [
+						"type": "playBgm",
+						"name": "Blood_Stain.opus",
+						"keep": true
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "耳旁再次响起象征着某个女孩子死去的声音。",
-				"bodyList": [
+						"type": "cgtextList",
+						"textList": "chapter000"
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "再一次——",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "0",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "再一次。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "1",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "女子们被肢解成单纯的肉块。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "2",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "我任由流下的血浸满全身,屏住自己的呼吸。",
-				"bodyList": [
+						"type": "changebg",
+						"img1": "other_0001.webp",
+						"memory1": false,
+						"img2": "bg_3563.webp",
+						"memory2": false,
+						"time": 30,
+						"style": "场景切换"
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "祈求自己能拥有从猎人手中逃脱的幸运。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "3",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3563.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "扑哧。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "4",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "bg_3563.webp",
-				"memory1": false,
-				"img2": "other_0002.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "直到刚才,我们还坐在去往娼馆的马车的路上。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "5",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "而在这之中的某些人,已经不在这个世上了。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "6",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "不,应该把“某些”换成“几乎所有”才更为恰当吧。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "7",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "恐怕,不久之后我也会变成小巷中血腥的装饰品。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "8",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "我是为了得到这种死法,才辛苦苟活至今的吗?",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "9",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "来个人告诉我啊——",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "10",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "谁都好。",
-				"bodyList": [
+						"type": "changebg",
+						"img1": "bg_3563.webp",
+						"memory1": false,
+						"img2": "other_0002.webp",
+						"memory2": false,
+						"time": 30,
+						"style": "场景切换"
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0002.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "来人啊!!",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "11",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "other_0002.webp",
-				"memory1": false,
-				"img2": "other_0001.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0001.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "少女",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy010000010.opus",
-				"text": "「呃······!?」",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "12",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "other_0001.webp",
-				"memory1": false,
-				"img2": "other_0003.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0003.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "漆黑的物体充斥了我的整个视野",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "13",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0003.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "我很快意识到,那是只很大的脚。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "14",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "other_0003.webp",
-				"memory1": false,
-				"img2": "",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "必须要出声求救。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "15",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "",
-				"memory1": false,
-				"img2": "other_0003.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0003.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "可是,耳中却只能听到自己的牙关不停交战的声音。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "16",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0003.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 50,
-				"wait": 1000,
-				"sound": "",
-				"text": "我是如此的无助。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "17",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "other_0003.webp",
-				"memory1": false,
-				"img2": "",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "逃跑也好,道歉也罢。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "18",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "就连抬头看一眼将要杀掉我的人的面孔都做不到。",
-				"bodyList": [
+						"type": "changebg",
+						"img1": "other_0002.webp",
+						"memory1": false,
+						"img2": "other_0001.webp",
+						"memory2": false,
+						"time": 30,
+						"style": "场景切换"
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "少女",
-				"time": 30,
-				"wait": 2000,
-				"sound": "aiy010000020.opus",
-				"text": "「······被杀」",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "19",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "会被杀。",
-				"bodyList": [
+						"type": "changebg",
+						"img1": "other_0001.webp",
+						"memory1": false,
+						"img2": "other_0003.webp",
+						"memory2": false,
+						"time": 30,
+						"style": "场景切换"
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "会被杀!!",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "20",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "来自内心深处的冰冷预感,渐渐地在体内蔓延开来。",
-				"bodyList": [
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "21",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "少女",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy010000030.opus",
-				"text": "「不,不要······」",
-				"bodyList": [
+						"type": "changebg",
+						"img1": "other_0003.webp",
+						"memory1": false,
+						"img2": "bg_0000.png",
+						"memory2": false,
+						"time": 30,
+						"style": "场景切换"
+					},
 					{
-						"name": "",
-						"px": 100,
-						"filter": false
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "22",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "changebg",
+						"img1": "bg_0000.png",
+						"memory1": false,
+						"img2": "other_0003.webp",
+						"memory2": false,
+						"time": 30,
+						"style": "场景切换"
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "23",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "24",
+						"time": 50,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "changebg",
+						"img1": "other_0003.webp",
+						"memory1": false,
+						"img2": "bg_0000.png",
+						"memory2": false,
+						"time": 30,
+						"style": "场景切换"
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "25",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "26",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "27",
+						"time": 30,
+						"wait": 2000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "28",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "29",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "30",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "cgtext",
+						"WindowSkin": false,
+						"head": {
+							"name": "",
+							"px": -300
+						},
+						"index": "31",
+						"time": 30,
+						"wait": 1000,
+						"bodyList": [
+							{
+								"name": "",
+								"px": 100,
+								"filter": false
+							}
+						]
+					},
+					{
+						"type": "changebg",
+						"img1": "bg_0000.png",
+						"memory1": false,
+						"img2": "",
+						"memory2": false,
+						"time": 30,
+						"style": "引出"
 					}
 				]
 			},
@@ -1206,2163 +1119,6 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 				"type": "playBgm",
 				"name": "Blind_Alley.opus",
 				"keep": true
-			},
-			{
-				"type": "changebg",
-				"img1": "",
-				"memory1": false,
-				"img2": "bg_6004.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_6004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "浮游都市,《诺瓦斯·艾蒂尔》。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_6004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "《特别受灾地区》——",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_6004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "通称,《牢狱》",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_6004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "是被险峻的峭壁环绕,与世隔绝的,都市的最底部。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_6004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000010.opus",
-				"text": "「放开我!」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_6004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000020.opus",
-				"text": "「我只是在帮那个女人而已!」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "bg_6004.webp",
-				"memory1": false,
-				"img2": "other_0004.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000030.opus",
-				"text": "「你们没听到吗!?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000040.opus",
-				"text": "「她是被受骗才会被卖到娼馆来的」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "other_0004.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000050.opus",
-				"text": "「用肮脏的手段把钱借给她父母的,就是你们这些家伙吧!?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "other_0004.webp",
-				"memory1": false,
-				"img2": "bg_3601.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000060.opus",
-				"text": "「给我说些什么啊」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000010.opus",
-				"text": "「这些话等到了娼馆再说吧」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000020.opus",
-				"text": "「我来抓你,只是受雇于人而已」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "bg_3601.webp",
-				"memory1": false,
-				"img2": "bg_3531.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "我走进娼馆《莉莉乌姆》的接待室。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "正在桌旁整理账簿的奥兹停下手头的工作,抬起头向我看来。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000010.opus",
-				"text": "「这不是凯伊姆先生吗,辛苦了」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000020.opus",
-				"text": "「委托已经完成了吗?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000030.opus",
-				"text": "「啊啊,是这家伙没错吧」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "奥兹用只要接触到就能杀人般的眼神在男人脸上搜过。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000030.opus",
-				"text": "「没错,就是这个人」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000040.opus",
-				"text": "「是么」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000070.opus",
-				"text": "「你,你们要对我做什么」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000040.opus",
-				"text": "「······」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "奥兹用一个眼神,就让男人闭上了嘴。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "然后,向我这边转过身来。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000050.opus",
-				"text": "「抱歉啊,总是麻烦你去做这些无聊的事」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000060.opus",
-				"text": "「都怪我们这边的年轻人太没用」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000050.opus",
-				"text": "「客套话就免了」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000070.opus",
-				"text": "「这还真是失礼了」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000080.opus",
-				"text": "「喂,来个人」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "光头男人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy820000010.opus",
-				"text": "「是」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000090.opus",
-				"text": "「凯伊姆先生做完工作回来了」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "光头男人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy820000020.opus",
-				"text": "「是,是,那个······」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000100.opus",
-				"text": "「我是要你拿些酒来,这个蠢材!」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "喀!",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "奥兹扔出的烟灰缸砸中了手下的额头。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "鲜血四溅。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000060.opus",
-				"text": "「不用这么麻烦」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000070.opus",
-				"text": "「我接下来要去《菲诺列塔》」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000110.opus",
-				"text": "「喔唷」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000120.opus",
-				"text": "「既然如此,我就不留您在这里喝难饮的劣质酒了」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "奥兹斜眼看着正捂住额头呻吟的手下,轻描淡写地说道。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000080.opus",
-				"text": "「用这些钱去买药」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "我将几枚铜钱仍在那个手下的身前。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000130.opus",
-				"text": "「凯伊姆先生,不用对他们这么好」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000090.opus",
-				"text": "「无妨」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000100.opus",
-				"text": "「话说回来,那个要落跑的女人呢?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000140.opus",
-				"text": "「我把她交给那些年轻人了,现在应该正在体会人生的严苛吧」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000150.opus",
-				"text": "「正好,趁此机会凯伊姆先生也来享受一番如何?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000080.opus",
-				"text": "「你,你们这些家伙,要对她做什么!?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "咣!",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "奥兹给了他一拳。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "一击即倒。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "喀,咚,咯!",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "奥兹毫不留情地向男人的脸上踩去。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000090.opus",
-				"text": "「咕······呃咳······」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "折断的牙齿伴着血泡被吐出。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "这份白色在鲜红色的液体中格外显眼。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000100.opus",
-				"text": "「你们以为做出这种事······卫兵会坐视不理吗······」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000160.opus",
-				"text": "「啊啊,不会坐视不理的」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000170.opus",
-				"text": "「应该会拿出你的钱包,和我们商量如何瓜分吧」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000110.opus",
-				"text": "「那,那种事······」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "这在牢狱是理所当然的事。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 2000,
-				"sound": "aiy350000180.opus",
-				"text": "「怎么,头一回来牢狱么?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "男人点了点头。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000190.opus",
-				"text": "「为了被骗的女人而来到牢狱,真是个规矩人啊」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000200.opus",
-				"text": "「······前提是,被骗的人不是你」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000120.opus",
-				"text": "「你说······我被骗了?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "年轻人",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy710000130.opus",
-				"text": "「那,那是怎么回事!?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000210.opus",
-				"text": "「不用急,今天晚上会好好告诉你的」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "奥兹抓起男人的脸。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "为引诱客人的怜悯之心而装纯,是娼妇的惯用手段。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "双亲被骗而借钱,结果作为抵押而将自己卖到这里,这是典型的说法。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "如果只是头脑发热而成为常客也就罢了,这次的男人热血过头,居然想出了要带女人私奔的计划。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "虽然女人半开玩笑地予以拒绝,但不知天高地厚的这家伙还是拉着她逃跑了。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "不过,想要逃脱追击本来就是不可能的任务。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "但即便如此,这种事情还是会一再的出现。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "说谎的女人和被骗的男人。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "在娼馆街,这是令人看到生厌的日常的风景。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000110.opus",
-				"text": "「我要走了」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000220.opus",
-				"text": "「好的,下次再麻烦您」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "奥兹",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy350000230.opus",
-				"text": "「之后吉克先生会将谢礼交给您的」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000120.opus",
-				"text": "「啊啊」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "我背向奥兹走出娼馆。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000130.opus",
-				"text": "「······?」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "bg_3531.webp",
-				"memory1": false,
-				"img2": "bg_3601.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "从远方传来微弱的歌声。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "是关卡广场的方向。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "对了。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "今天有觐见圣女的仪式。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "当代的圣女伊莲——",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "俗称《盲眼之圣女》,据说即使在历代的圣女中,人气也是数一数二的。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "广场上的人估计相当多吧。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "虽然我也想去看看她长什么样,不过要在人潮中挤来挤去就免了。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "还是老老实实去菲诺列塔喝烧酒吧。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "",
-				"time": 30,
-				"wait": 1000,
-				"sound": "",
-				"text": "正当我这样想着的时候,一个身影自小巷的那头走来。",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "bg_3601.webp",
-				"memory1": false,
-				"img2": "bg_3531.webp",
-				"memory2": false,
-				"time": 30,
-				"style": "场景切换"
-			},
-			{
-				"type": "cgtext",
-				"bg": "bg_3531.webp",
-				"memory": false,
-				"WindowSkin": false,
-				"head": {
-					"name": "",
-					"px": -300
-				},
-				"name": "凯伊姆",
-				"time": 30,
-				"wait": 1000,
-				"sound": "aiy310000140.opus",
-				"text": "「艾莉斯」",
-				"bodyList": [
-					{
-						"name": "",
-						"px": 100,
-						"filter": false
-					}
-				]
-			},
-			{
-				"type": "changebg",
-				"img1": "bg_3531.webp",
-				"memory1": false,
-				"img2": "",
-				"memory2": false,
-				"time": 30,
-				"style": "引出"
-			},
-			{
-				"type": "comment",
-				"text": "好多立绘,先更这点(吐槽一下)"
 			}
 		],
 		"chapter01": [
@@ -3375,20 +1131,20 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 				"time": 30,
 				"style": "引入"
 			},
+			{
+				"type": "cgtextList",
+				"textList": "chapter01"
+			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020106.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "0",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000005.opus",
-				"text": "「啊,凯伊姆」",
 				"bodyList": [
 					{
 						"name": "",
@@ -3397,20 +1153,20 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 					}
 				]
 			},
+			{
+				"type": "playBgm",
+				"name": "Halbmond.opus"
+			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020141.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "1",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000010.opus",
-				"text": "「正好,我还想要去找你呢」",
 				"bodyList": [
 					{
 						"name": "tati_020141.webp",
@@ -3421,18 +1177,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020157.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "2",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000020.opus",
-				"text": "「没想到凯伊姆会主动出现······这是命运吗?」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3443,18 +1195,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "3",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000150.opus",
-				"text": "「显然不是吧」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3465,18 +1213,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020107.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "4",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000030.opus",
-				"text": "「啊,是么」",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -3487,18 +1231,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "5",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "艾莉斯挑了挑整齐的双眉,微微地哼了一声。",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -3509,18 +1249,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "6",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "虽然是个相当引人注目的美人,但她这个将亲切儿子丢入无底深渊的性格,为自己扣了不少的分",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -3531,18 +1267,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "7",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "给人印象最深的,就是那潭水般的双瞳。",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -3553,18 +1285,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "8",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "在漆黑的瞳孔中,完全看不出感情的波动。",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -3575,18 +1303,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020157.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "9",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000040.opus",
-				"text": "「喜欢我的眼睛吗?」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3597,18 +1321,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020157.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "10",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000050.opus",
-				"text": "「如果想要的话就给你吧?」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3619,18 +1339,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "11",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000160.opus",
-				"text": "「用不着」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3641,18 +1357,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020111.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "12",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000060.opus",
-				"text": "「阿拉,可惜」",
 				"bodyList": [
 					{
 						"name": "tati_020111.webp",
@@ -3663,18 +1375,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "13",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000170.opus",
-				"text": "「那么,找我有什么事」",
 				"bodyList": [
 					{
 						"name": "tati_020111.webp",
@@ -3685,18 +1393,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020101.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "14",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000070.opus",
-				"text": "「梅尔特的钱好像被偷了」",
 				"bodyList": [
 					{
 						"name": "tati_020101.webp",
@@ -3707,18 +1411,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "15",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000180.opus",
-				"text": "「钱被偷了?都几岁了还这么没用」",
 				"bodyList": [
 					{
 						"name": "tati_020101.webp",
@@ -3729,18 +1429,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020107.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "16",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000080.opus",
-				"text": "「不要对我说啊」",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -3751,18 +1447,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "17",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000190.opus",
-				"text": "「那家伙,该不会说要让我去抓那个小偷吧?」",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -3773,18 +1465,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020157.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "18",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000090.opus",
-				"text": "「就是这样」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3795,18 +1483,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "19",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000200.opus",
-				"text": "「笨蛋吗」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3817,18 +1501,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "20",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000210.opus",
-				"text": "「如果是小钱的话,就当做是买个教训吧」",
 				"bodyList": [
 					{
 						"name": "tati_020157.webp",
@@ -3839,18 +1519,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020125.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "21",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000100.opus",
-				"text": "「说起来,被盗的是这个月的上纳金」",
 				"bodyList": [
 					{
 						"name": "tati_020125.webp",
@@ -3861,18 +1537,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "22",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000220.opus",
-				"text": "「你说什么?」",
 				"bodyList": [
 					{
 						"name": "tati_020125.webp",
@@ -3883,18 +1555,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020121.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "23",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000110.opus",
-				"text": "「用这些钱买教训,也太过奢侈了呢」",
 				"bodyList": [
 					{
 						"name": "tati_020121.webp",
@@ -3905,18 +1573,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "24",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000230.opus",
-				"text": "「知道了,我去找」",
 				"bodyList": [
 					{
 						"name": "tati_020121.webp",
@@ -3927,18 +1591,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "25",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000240.opus",
-				"text": "「小偷的特征呢」",
 				"bodyList": [
 					{
 						"name": "tati_020121.webp",
@@ -3949,18 +1609,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020105.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "26",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000120.opus",
-				"text": "「男孩子」",
 				"bodyList": [
 					{
 						"name": "tati_020105.webp",
@@ -3971,18 +1627,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020103.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "27",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000130.opus",
-				"text": "「······而且,背后有翅膀」",
 				"bodyList": [
 					{
 						"name": "tati_020103.webp",
@@ -3993,18 +1645,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020141.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "28",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000140.opus",
-				"text": "「虽然姑且是藏在身后,但是仔细观察的话是很明显的」",
 				"bodyList": [
 					{
 						"name": "tati_020141.webp",
@@ -4015,18 +1663,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "29",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000250.opus",
-				"text": "「羽化病吗」",
 				"bodyList": [
 					{
 						"name": "tati_020141.webp",
@@ -4037,18 +1681,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020105.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "30",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000150.opus",
-				"text": "「那些人可是毫不留情的,所以即使是为了那个孩子,也要赶快抓到他」",
 				"bodyList": [
 					{
 						"name": "tati_020105.webp",
@@ -4059,18 +1699,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "31",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000260.opus",
-				"text": "「注意到他逃窜的方向了吗?」",
 				"bodyList": [
 					{
 						"name": "tati_020105.webp",
@@ -4081,18 +1717,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020105.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "32",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000160.opus",
-				"text": "「广场那边」",
 				"bodyList": [
 					{
 						"name": "tati_020105.webp",
@@ -4103,18 +1735,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020107.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "33",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000170.opus",
-				"text": "「虽然刚才《不蚀金锁》的人去追了,不过多半是······」",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -4125,18 +1753,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "34",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000280.opus",
-				"text": "「偏偏还是广场吗」",
 				"bodyList": [
 					{
 						"name": "tati_020107.webp",
@@ -4147,18 +1771,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_020141.webp",
 					"px": -300
 				},
-				"name": "艾莉斯",
+				"index": "35",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy020000180.opus",
-				"text": "「今天是觐见圣女大人的日子」",
 				"bodyList": [
 					{
 						"name": "tati_020145.webp",
@@ -4169,18 +1789,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "36",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000290.opus",
-				"text": "「我知道」",
 				"bodyList": [
 					{
 						"name": "tati_020145.webp",
@@ -4191,18 +1807,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3601.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "37",
 				"time": 30,
 				"wait": 1000,
-				"sound": "aiy310000300.opus",
-				"text": "「尽量找找看就好」",
 				"bodyList": [
 					{
 						"name": "tati_020145.webp",
@@ -4231,20 +1843,20 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 				"time": 30,
 				"style": "引入"
 			},
+			{
+				"type": "cgtextList",
+				"textList": "chapter02"
+			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "不蚀金锁成员",
+				"index": "0",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「凯伊姆先生,凯伊姆先生」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4255,18 +1867,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "不蚀金锁成员",
+				"index": "1",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「您已经和艾莉斯大夫见过面了吗?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4277,18 +1885,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "2",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「啊啊,所以才会追过来的」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4299,18 +1903,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "3",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「看到小偷了吗?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4321,18 +1921,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "不蚀金锁成员",
+				"index": "4",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「没有,他向广场那边逃了过去,今天这么拥挤,我们也只能放弃了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4343,18 +1939,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "不蚀金锁成员",
+				"index": "5",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不过,我也只是刚好在店里所以才追了过去,并不是受人所托」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4365,18 +1957,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "不蚀金锁成员",
+				"index": "6",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我已经准备撤退了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4387,18 +1975,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "不蚀金锁成员",
+				"index": "7",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「凯伊姆先生还要继续追吗?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4409,18 +1993,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "8",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「啊啊」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4431,18 +2011,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3561.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "9",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "做完情报交换之后,我跟男人道别。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4462,18 +2038,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "10",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「和我想的一样啊······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4484,18 +2056,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "11",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "在牢狱中最大的广场上,聚集着看不到尽头的人群。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4506,18 +2074,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "12",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "就算是来参见圣女祈祷,这人数也太多了点吧。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4528,18 +2092,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "13",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "自然,我也找不到逃跑的孩子。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4550,18 +2110,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "14",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "是混杂到人群中了吧。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4572,18 +2128,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "15",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "如果已经从广场上脱身了的话,就更难发现了。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4594,18 +2146,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "16",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "只好赌他还在这里了。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4625,18 +2173,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_0000.png",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "17",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "我先移动到了一个视野良好的地方。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4647,18 +2191,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_0000.png",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "18",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "从这里,一眼就可以看到人群的变化。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4669,18 +2209,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_0000.png",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "19",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "广场还是沸腾起来。",
 				"bodyList": [
 					{
 						"name": "",
@@ -4704,18 +2240,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "20",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "抬头望去,原来是在天台之上出现了一个人影",
 				"bodyList": [
 					{
 						"name": "tati_440101.webp",
@@ -4726,18 +2258,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "21",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "但是,与周围的期待不同,现身的是一名中年的神官。",
 				"bodyList": [
 					{
 						"name": "tati_440101.webp",
@@ -4748,18 +2276,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "22",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "骂声四溢。",
 				"bodyList": [
 					{
 						"name": "tati_440101.webp",
@@ -4770,18 +2294,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "23",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "神官则是笑着摆正衣领",
 				"bodyList": [
 					{
 						"name": "",
@@ -4792,18 +2312,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440101.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "24",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「从现在开始,举行谒见的仪式」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4814,18 +2330,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440105.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "25",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「在参见那位大人之前,我希望牢狱的诸位再次思考这个《诺瓦斯·艾蒂尔》存在的意义······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4836,18 +2348,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440101.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "26",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「初代圣女伊莲大人,便是也难怪这崇高的祈祷之力,令《诺瓦斯·艾蒂尔》浮在空中,拯救了我们的祖先」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4858,18 +2366,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440110.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "27",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「这之后的几百年来,传承了初代大人力量的历代圣女伊莲大人,让这里留在了空中」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4880,18 +2384,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440101.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "28",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「这座都市是被圣女大人守护的人类最后的圣域,而我们则是被选召的虔诚的信徒」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4902,18 +2402,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440102.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "29",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「怀着对圣女的感激祈祷吧,感谢圣女伊莲吧!并献上祈祷!」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4950,18 +2446,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030102.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "30",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不忘感谢与祈祷,神才会拯救我们」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4972,18 +2464,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030101.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "31",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「与我一起,向审虔诚地祈祷吧」",
 				"bodyList": [
 					{
 						"name": "",
@@ -4998,18 +2486,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "32",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "广场上欢声雷动。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5020,18 +2504,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "33",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "圣女没有回应喧嚣的人声,而是静静地合上双眼面向广场。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5042,18 +2522,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "34",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "虽然感觉有些冷淡,但总比像个傻瓜似的笑着向这群人挥手要强。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5064,18 +2540,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "35",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "她掌握着这条街道,还有在这条街上生活的人的命运。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5086,18 +2558,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "36",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "比起揽得人气,她更想要为了街道的继续存在而献出全力。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5108,18 +2576,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "37",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "也是为了不让《大崩落》的惨剧再度发生。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5130,18 +2594,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "38",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "十几年前的那场悲剧。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5152,18 +2612,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "39",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "虽然在我脑海中的记忆已经相当模糊,但哪怕只是稍有触及,不快的感觉都会在胸口蔓延开。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5174,18 +2630,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "40",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5196,18 +2648,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "41",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "这时我才想起,现在不是我在这里看圣女的时候。",
 				"bodyList": [
 					{
 						"name": "",
@@ -5222,18 +2670,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "eve_030101.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "女声",
+				"index": "42",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「——っ!?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5253,18 +2697,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "围观的女人",
+				"index": "43",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「羽,羽化病人!?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5275,18 +2715,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "围观的中年人",
+				"index": "44",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「喂,谁去叫下羽狩」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5297,18 +2733,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "惊慌的观众",
+				"index": "45",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你这家伙不要靠近我,要是传染了可怎么办」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5319,18 +2751,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "粗鲁的观众",
+				"index": "46",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你这小鬼赶快滚开」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5341,18 +2769,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3021.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "47",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「接下来」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5372,18 +2796,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030107.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "48",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「发生什么事了?看上去似乎很嘈杂」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5394,18 +2814,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_130104.webp",
 					"px": -300
 				},
-				"name": "随从",
+				"index": "49",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「似乎是某个人逃跑了······具体的我也不是很清楚」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5416,18 +2832,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440103.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "50",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「圣女大人,继续待在天台上可能会出事,请您先回到室内吧」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5438,18 +2850,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030145.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "51",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不用在意我,比起那个,我更关心究竟发生了什么事」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5460,18 +2868,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440107.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "52",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「对不起,我真的不知道」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5482,18 +2886,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030145.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "53",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······是吗」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5504,18 +2904,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320101.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "54",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「恕我僭越,请准许我说明情况」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5526,18 +2922,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320105.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "55",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「在来觐见的人群中出现了《羽化病》的患者」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5548,18 +2940,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320101.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "56",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「围观的人群因而产生了骚动」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5570,18 +2958,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320102.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "57",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「现在,《防疫局》已经派遣了部队。我想不久之后,他们就会安静下来了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5592,18 +2976,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030124.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "58",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「羽化病······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5614,18 +2994,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320107.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "59",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「怎么了?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5636,18 +3012,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030125.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "60",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「没什么」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5658,18 +3030,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030141.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "61",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「辛苦了,你的名字是?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5680,18 +3048,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320102.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "62",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「属下是在防疫局任职的,鲁基乌斯· 迪斯·米利尤」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5702,18 +3066,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440102.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "63",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「噢噢,阁下就是鲁基乌斯卿吗,我听说过你的传闻」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5724,18 +3084,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440109.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "64",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「阁下是在工作上相当出色的人呢」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5746,18 +3102,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320101.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "65",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不敢当」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5768,18 +3120,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320108.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "66",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「话说回来,这次是属下警备工作的失职。让圣女大人见到这不成体统的一面,请您见谅」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5790,18 +3138,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030145.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "67",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「即使是目不见物的我,也能感受到聚集于此的民众数量之多。警备工作难以展开也在情理之中」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5812,18 +3156,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320101.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "68",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「属下不胜惶恐」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5834,18 +3174,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320102.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "69",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「接下来属下还要回到工作岗位上,在这里就先告退了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5856,18 +3192,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030101.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "70",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「鲁基乌斯先生」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5878,18 +3210,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320101.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "71",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「属下在」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5900,18 +3228,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030105.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "72",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你是怎样看待羽狩的工作的呢?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5922,18 +3246,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440106.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "73",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「圣,圣女大人」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5944,18 +3264,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320101.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "74",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「防疫局的工作是国王陛下赐予的重要职务。属下非常荣幸能够为这个都市的繁荣尽一份微薄之力」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5966,18 +3282,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_440102.webp",
 					"px": -300
 				},
-				"name": "神官",
+				"index": "75",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不,不亏是鲁基乌斯卿,相当优秀的想法」",
 				"bodyList": [
 					{
 						"name": "",
@@ -5988,18 +3300,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_030125.webp",
 					"px": -300
 				},
-				"name": "圣女",
+				"index": "76",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「是吗。辛苦你了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6010,18 +3318,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_130124.webp",
 					"px": -300
 				},
-				"name": "随从",
+				"index": "77",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「圣女大人······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6032,18 +3336,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320102.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "78",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6054,18 +3354,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3026.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_320101.webp",
 					"px": -300
 				},
-				"name": "鲁基乌斯",
+				"index": "79",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那么,属下就回岗位去了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6082,10 +3378,6 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 				"memory2": false,
 				"time": 30,
 				"style": "引出"
-			},
-			{
-				"type": "comment",
-				"text": "下一场景是追捕羽化病少年...做个标记"
 			}
 		],
 		"chapter03": [
@@ -6098,20 +3390,20 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 				"time": 30,
 				"style": "引入"
 			},
+			{
+				"type": "cgtextList",
+				"textList": "chapter03"
+			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "0",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "从羽化病的少年纷乱的足音中,可以听得出相当的疲劳。",
 				"bodyList": [
 					{
 						"name": "",
@@ -6122,18 +3414,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "1",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "显然,他并没有想到我会捷足先登吧。",
 				"bodyList": [
 					{
 						"name": "",
@@ -6144,18 +3432,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "2",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "少年惶恐地回头看了一眼后,微微露出安心的表情,双手拄在膝盖上。",
 				"bodyList": [
 					{
 						"name": "",
@@ -6166,18 +3450,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "3",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「辛苦你了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6188,18 +3468,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "4",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「稀!?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6210,18 +3486,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "5",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「逃到贫民区是个不错的想法」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6232,18 +3504,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "6",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你,你是,羽狩吗?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6254,18 +3522,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "7",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不是」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6276,18 +3540,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "8",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「什,什么啊······混蛋,不要吓我啊」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6298,18 +3558,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "9",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我对令你受惊这件事致以歉意」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6320,18 +3576,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "10",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「作为回报,麻烦你把从店里偷的钱交出来吧」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6342,18 +3594,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "11",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「钱?你在说什么」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6364,18 +3612,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "12",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你要找的腰上的东西,掉在你身后了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6386,18 +3630,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "13",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「哎?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6412,18 +3652,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "14",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「呃呀」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6434,18 +3670,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "15",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你······你这混蛋」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6456,18 +3688,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "16",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6478,18 +3706,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "17",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「把偷的钱交出来」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6500,18 +3724,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "18",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我不知道你在······咕」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6530,18 +3750,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "19",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你,你说是我偷的······有什么证据吗」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6552,18 +3768,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "20",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你还挺倔的啊」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6574,18 +3786,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "21",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不过,给我听好了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6596,18 +3804,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "22",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你偷的那些钱,是要上缴给《不蚀金锁》的上纳金」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6618,18 +3822,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "23",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「而且,钱的主人是从前和吉克颇有渊源的女人」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6640,18 +3840,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "24",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「吉克?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6662,18 +3858,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "25",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「他是《不蚀金锁》的主人,这么说你就明白了吧」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6684,18 +3876,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "26",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「哎?哎?怎么会······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6706,18 +3894,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "27",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「再问你一遍,钱在哪里?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6728,18 +3912,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "28",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「是,是,是,在我的怀里」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6750,18 +3930,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "29",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你没有擅自拿掉一部分吧」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6772,18 +3948,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "30",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「是,是的」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6794,18 +3966,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "31",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那,那个,您是《不蚀金锁》的人吗?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6816,18 +3984,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "32",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「算是吧」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6838,18 +4002,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "33",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我什么都可以做,请您一定要帮帮我」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6860,18 +4020,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "34",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「抱歉,我并没有兴趣去帮助他人」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6882,18 +4038,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "35",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我什么······什么,都会做的······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6904,18 +4056,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "36",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我一直都是生活在下层的」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6926,18 +4074,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "37",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「可是,不知何时染上了羽化病······背后长出了翅膀······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6948,18 +4092,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "38",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「被寄宿工作的店赶了出来,只得流落到牢狱这里」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6970,18 +4110,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "39",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「因为独自实在是饿的不行了,所以才会偷这些钱的」",
 				"bodyList": [
 					{
 						"name": "",
@@ -6992,18 +4128,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "40",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我明明没有做任何坏事······为什么······会遇到这种事······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7014,18 +4146,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "41",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「谁知道」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7036,18 +4164,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "42",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「呜······呜呜······接下来,要对我做什么?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7058,18 +4182,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "43",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我要把你带到组织那里」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7080,18 +4200,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "44",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「怎,怎么这样」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7102,18 +4218,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "45",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不过,那样做的前提是你不是羽化病人」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7124,18 +4236,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "46",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「组织也没有笨到把羽化病人招待到家里的程度」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7146,18 +4254,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "47",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那么,是要放我逃走吗?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7168,18 +4272,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "48",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我要让你学到教训」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7194,18 +4294,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "49",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「如果换做是组织的制裁,至少要有断条胳膊的觉悟」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7216,18 +4312,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "50",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你的运气不错」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7238,18 +4330,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "51",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「唔······啊,是的······」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7260,18 +4348,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "52",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「滚」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7282,18 +4366,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "53",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「非常感谢」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7304,18 +4384,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "54",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「唔啊!?」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7330,18 +4406,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430103.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "55",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「到这里就结束了,羽化病人」",
 				"bodyList": [
 					{
 						"name": "tati_430103.webp",
@@ -7352,18 +4424,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "男",
+				"index": "56",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「确认他的翅膀」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7374,18 +4442,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "57",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "趁还没有被卷入麻烦的事情之前,赶快离开这里吧",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7414,18 +4478,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430106.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "58",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那边的那个人」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7436,18 +4496,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "59",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······有什么事?」",
 				"bodyList": [
 					{
 						"name": "tati_430101.webp",
@@ -7458,18 +4514,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430101.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "60",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「可以稍微让我问几句话吗」",
 				"bodyList": [
 					{
 						"name": "tati_430101.webp",
@@ -7480,18 +4532,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "61",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······」",
 				"bodyList": [
 					{
 						"name": "tati_430101.webp",
@@ -7502,18 +4550,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "62",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「啊啊,无妨」",
 				"bodyList": [
 					{
 						"name": "tati_430101.webp",
@@ -7524,18 +4568,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "63",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「感谢您的合作」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7546,18 +4586,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "",
+				"index": "64",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "队长殷勤地致以谢礼。",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7568,18 +4604,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "",
+				"index": "65",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "而在他的眼前,少年的衣服已经被他的补下们扯破。",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7590,18 +4622,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "",
+				"index": "66",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "在瘦骨嶙峋的裸露后背上,长有纯白的羽翼。",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7612,18 +4640,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "红发的羽狩",
+				"index": "67",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「副队长,确认翅膀的持有了」",
 				"bodyList": [
 					{
 						"name": "",
@@ -7634,18 +4658,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "68",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「保护他」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7656,18 +4676,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "69",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「不要······请原谅,我······」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7678,18 +4694,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "70",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我们只是要带你去治愈院治疗羽化病,不是什么应该感到害怕的事情」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7700,18 +4712,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "71",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「可是,可是」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7722,18 +4730,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "72",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「没关系的」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7744,18 +4748,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "73",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······哥,哥哥」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7766,18 +4766,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "74",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你是羽化病人的亲属吗?」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7788,18 +4784,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "75",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「只是路人而已」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7810,18 +4802,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "76",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「顺带一提,我也没有打算找你们的麻烦」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7832,18 +4820,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430101.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "77",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「前几天,有个和你说了同样的话的人,在我们背向他的瞬间对我们发动了袭击」",
 				"bodyList": [
 					{
 						"name": "tati_430101.webp",
@@ -7854,18 +4838,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430108.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "78",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我的一个部下就此永久失去了半截胳膊」",
 				"bodyList": [
 					{
 						"name": "tati_430108.webp",
@@ -7876,18 +4856,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "79",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我表示同情」",
 				"bodyList": [
 					{
 						"name": "tati_430108.webp",
@@ -7898,18 +4874,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "80",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我马上就会消失的,这样就没问题了吧?」",
 				"bodyList": [
 					{
 						"name": "tati_430108.webp",
@@ -7920,18 +4892,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "81",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「嘛,不要这么慌张」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7942,18 +4910,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "82",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "副队长看着羽化的少年。",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -7964,18 +4928,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "83",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你与这个人是什么关系?没有被他殴打吗?」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -7986,18 +4946,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "84",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「没,没有」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8008,18 +4964,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "85",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「如何对我们保持合作,你就可以在治愈院得到优先的治疗」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8030,18 +4982,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "86",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8052,18 +5000,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "87",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那个人,是《不蚀金锁》的组织成员······」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8074,18 +5018,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "羽化病患少年",
+				"index": "88",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「突然说让我拿出钱来,我刚一拒绝他就打我」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8096,18 +5036,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "89",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「原来如此······」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8118,18 +5054,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430107.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "90",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那位少年说你是《不蚀金锁》的一员,不知此事是否属实?」",
 				"bodyList": [
 					{
 						"name": "tati_430107.webp",
@@ -8140,18 +5072,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "91",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「当然不是」",
 				"bodyList": [
 					{
 						"name": "tati_430107.webp",
@@ -8162,18 +5090,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "92",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我只是从那里接受工作而已,并不是他们的成员」",
 				"bodyList": [
 					{
 						"name": "tati_430107.webp",
@@ -8184,18 +5108,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "93",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你的意思是说,少年在说谎吗?」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8206,18 +5126,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "94",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「啊啊」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8228,18 +5144,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "95",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「如果你们和组织有关系的话,只要问问我是不是那里的成员就能明白事实了吧」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8250,18 +5162,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430107.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "96",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「就算我去询问,也无法从他们那里得到事实」",
 				"bodyList": [
 					{
 						"name": "tati_430107.webp",
@@ -8272,18 +5180,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430108.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "97",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「《不蚀金锁》的那些人一向都不对我们合作,我对此深感困扰」",
 				"bodyList": [
 					{
 						"name": "tati_430108.webp",
@@ -8294,18 +5198,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "98",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「真是辛苦啊」",
 				"bodyList": [
 					{
 						"name": "tati_430108.webp",
@@ -8316,18 +5216,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "99",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「说的是啊」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8338,18 +5234,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "100",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「其实,砍下我部下胳膊的似乎也是组织的成员呢」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8360,18 +5252,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "101",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「无需如此警戒,我只是想在看守所向你咨询一些事情而已」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8382,18 +5270,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "102",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「如果能够知晓牢狱与组织的事情,我们也可以尽可能地对更多的羽化病人进行保护」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8404,18 +5288,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430102.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "103",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那和整条街道的和平也是紧密相关的吧?」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8426,18 +5306,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "104",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我知道,你们有逮捕干扰狩猎羽化病人的权力」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8448,18 +5324,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "105",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「但是,我没有对你们做出任何干扰,为什么要对我如此纠缠不休呢」",
 				"bodyList": [
 					{
 						"name": "tati_430102.webp",
@@ -8470,18 +5342,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "106",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那些话,我们会在看守所对你详细说明的」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8492,18 +5360,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "107",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8514,18 +5378,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "108",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "在这里起争执的话,就会被羽狩加害。",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8536,18 +5396,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "109",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "就算能从这里脱身,今后只要碰面就会产生纠纷也是摆明的事情。",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8558,18 +5414,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "110",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "就算逃跑,也没有好的结果。",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8580,18 +5432,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "111",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "正当我想要打圆场的时候,刚才的气氛一瞬间产生了转变。",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8602,18 +5450,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "112",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "发生了什么事······",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8624,18 +5468,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050101.webp",
 					"px": -300
 				},
-				"name": "??",
+				"index": "113",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我认为,那位先生是正确的」",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8646,18 +5486,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "114",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "耳边突然响起了一个能够将这阴暗的气息一扫而空的声音",
 				"bodyList": [
 					{
 						"name": "tati_430105.webp",
@@ -8668,18 +5504,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "115",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "羽狩们一起回头。",
 				"bodyList": [
 					{
 						"name": "",
@@ -8690,18 +5522,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "116",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "而在他们视线的焦点处,",
 				"bodyList": [
 					{
 						"name": "",
@@ -8712,18 +5540,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "117",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "伫立着一位女性。",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -8734,18 +5558,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "118",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "在端正的容颜下,代表着强烈意志的双眉十分显眼。",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -8756,18 +5576,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "119",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "身体的柔软与紧紧包裹在其身上的御兽支付,两者显得十分的不搭配。",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -8778,18 +5594,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "120",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "我还是第一次看到女性的羽狩。",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -8800,18 +5612,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "羽狩的副队长",
+				"index": "121",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「队长,这是获得《不蚀金锁》情报的好机会」",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -8827,18 +5635,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430141.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "122",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「兰格副队长,就算是为了获得情报,也不能做出恫吓的发言啊」",
 				"bodyList": [
 					{
 						"name": "tati_050141.webp",
@@ -8854,18 +5658,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430106.webp",
 					"px": -300
 				},
-				"name": "兰格副队长",
+				"index": "123",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我并没有打算去恫吓他······」",
 				"bodyList": [
 					{
 						"name": "tati_050141.webp",
@@ -8881,18 +5681,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050221.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "124",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「告诉我那个被砍掉胳膊的队员的名字」",
 				"bodyList": [
 					{
 						"name": "tati_050221.webp",
@@ -8908,18 +5704,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050223.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "125",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我会去探望他的」",
 				"bodyList": [
 					{
 						"name": "tati_050223.webp",
@@ -8935,18 +5727,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430104.webp",
 					"px": -300
 				},
-				"name": "兰格副队长",
+				"index": "126",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「那个是······」",
 				"bodyList": [
 					{
 						"name": "tati_050223.webp",
@@ -8962,18 +5750,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050101.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "127",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我知道,你一直在为有所收获而努力工作」",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -8989,18 +5773,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050105.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "128",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「但是,正因为我们的工作是为民众提供帮助」",
 				"bodyList": [
 					{
 						"name": "tati_050105.webp",
@@ -9016,18 +5796,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050103.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "129",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「所以就更不能损害人与人之间的信赖」",
 				"bodyList": [
 					{
 						"name": "tati_050103.webp",
@@ -9043,18 +5819,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_430105.webp",
 					"px": -300
 				},
-				"name": "兰格副队长",
+				"index": "130",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我会铭记在心」",
 				"bodyList": [
 					{
 						"name": "tati_050103.webp",
@@ -9070,18 +5842,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050101.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "131",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「这位先生,我的部下失礼了」",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -9092,18 +5860,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "132",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「只要不对我再来一次就好」",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -9114,18 +5878,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050126.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "133",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「请稍等」",
 				"bodyList": [
 					{
 						"name": "tati_050126.webp",
@@ -9136,18 +5896,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "134",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「有什么事?」",
 				"bodyList": [
 					{
 						"name": "tati_050126.webp",
@@ -9158,18 +5914,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050101.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "135",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我想确认一件事」",
 				"bodyList": [
 					{
 						"name": "tati_050101.webp",
@@ -9180,18 +5932,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050105.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "136",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「你真的不是《不蚀金锁》的成员吗?」",
 				"bodyList": [
 					{
 						"name": "tati_050105.webp",
@@ -9202,18 +5950,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "137",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「真的」",
 				"bodyList": [
 					{
 						"name": "tati_050105.webp",
@@ -9224,18 +5968,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "138",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「如果我说是的话,你有什么打算?」",
 				"bodyList": [
 					{
 						"name": "tati_050105.webp",
@@ -9246,18 +5986,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_050143.webp",
 					"px": -300
 				},
-				"name": "羽狩的队长",
+				"index": "139",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「我听过传闻,说他们是用依靠暴力而得的钱在生活」",
 				"bodyList": [
 					{
 						"name": "tati_050143.webp",
@@ -9268,18 +6004,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "140",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「······这样啊」",
 				"bodyList": [
 					{
 						"name": "tati_050143.webp",
@@ -9290,18 +6022,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3801.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "141",
 				"time": 30,
 				"wait": 1000,
-				"sound": "",
-				"text": "「如果能有收获就好了啊」",
 				"bodyList": [
 					{
 						"name": "tati_050143.webp",
@@ -9357,20 +6085,20 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 				"time": 30,
 				"style": "场景切换"
 			},
+			{
+				"type": "cgtextList",
+				"textList": "chapter04"
+			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120122.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "0",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「欢迎光临」",
 				"bodyList": [
 					{
 						"name": "",
@@ -9381,18 +6109,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120101.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "1",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「辛苦了」",
 				"bodyList": [
 					{
 						"name": "tati_120101.webp",
@@ -9403,18 +6127,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120107.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "2",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「抱歉,又拜托给你了个这么麻烦的工作」",
 				"bodyList": [
 					{
 						"name": "tati_120107.webp",
@@ -9425,18 +6145,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "3",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「没什么,比想象中完成的更容易」",
 				"bodyList": [
 					{
 						"name": "tati_120107.webp",
@@ -9447,18 +6163,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120112.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "4",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「那就好」",
 				"bodyList": [
 					{
 						"name": "tati_120112.webp",
@@ -9469,18 +6181,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120102.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "5",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「这是我的一点谢意」",
 				"bodyList": [
 					{
 						"name": "tati_120102.webp",
@@ -9491,18 +6199,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "6",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「味道有些变化啊」",
 				"bodyList": [
 					{
 						"name": "tati_120102.webp",
@@ -9513,18 +6217,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120107.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "7",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「啊,被发现了?」",
 				"bodyList": [
 					{
 						"name": "tati_120107.webp",
@@ -9535,18 +6235,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120111.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "8",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「最近,没能到手什么好的原料呢」",
 				"bodyList": [
 					{
 						"name": "tati_120111.webp",
@@ -9557,18 +6253,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "9",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「去拜托吉克如何?」",
 				"bodyList": [
 					{
 						"name": "tati_120111.webp",
@@ -9579,18 +6271,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120127.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "10",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「话是这么说,但是总不能用店里采购的这种小事去麻烦他吧······」",
 				"bodyList": [
 					{
 						"name": "tati_120127.webp",
@@ -9601,18 +6289,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "11",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「那希望你也不要来麻烦我」",
 				"bodyList": [
 					{
 						"name": "tati_120127.webp",
@@ -9623,18 +6307,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120112.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "12",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「那 是 两 码 事」",
 				"bodyList": [
 					{
 						"name": "tati_120112.webp",
@@ -9645,18 +6325,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120113.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "13",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「再说,凯伊姆是靠着工作来生活的吧」",
 				"bodyList": [
 					{
 						"name": "tati_120113.webp",
@@ -9667,18 +6343,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120113.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "14",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「而且,自己的钱被偷了这么害羞的事,向凯伊姆意外的其他人都说不出口」",
 				"bodyList": [
 					{
 						"name": "tati_120113.webp",
@@ -9689,18 +6361,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "15",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「反正,也已经传到吉克的耳朵里了」",
 				"bodyList": [
 					{
 						"name": "tati_120113.webp",
@@ -9711,18 +6379,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120112.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "16",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「这是面子问题啊,面子问题」",
 				"bodyList": [
 					{
 						"name": "tati_120112.webp",
@@ -9733,18 +6397,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "17",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「嘛,算了」",
 				"bodyList": [
 					{
 						"name": "tati_120101.webp",
@@ -9755,18 +6415,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "18",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「这样就好了吧?」",
 				"bodyList": [
 					{
 						"name": "tati_120101.webp",
@@ -9777,18 +6433,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120121.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "19",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「这是钱包呢」",
 				"bodyList": [
 					{
 						"name": "tati_120121.webp",
@@ -9799,18 +6451,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120102.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "20",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「嗯,东西没少」",
 				"bodyList": [
 					{
 						"name": "tati_120102.webp",
@@ -9821,18 +6469,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120111.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "21",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「太好啦—这个月的上纳金,我可全部都放在里面了呢」",
 				"bodyList": [
 					{
 						"name": "tati_120111.webp",
@@ -9843,18 +6487,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120113.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "22",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「如果没有找到的话,说不定就又会被送到娼馆里了呢」",
 				"bodyList": [
 					{
 						"name": "tati_120113.webp",
@@ -9865,18 +6505,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "23",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「在那边不是来钱更快吗?」",
 				"bodyList": [
 					{
 						"name": "tati_120113.webp",
@@ -9887,18 +6523,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120106.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "24",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「阿拉,你是在说我还能有魅力吗?」",
 				"bodyList": [
 					{
 						"name": "tati_120106.webp",
@@ -9909,18 +6541,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "25",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「这是客套话而已」",
 				"bodyList": [
 					{
 						"name": "tati_120106.webp",
@@ -9931,18 +6559,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120113.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "26",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「欺负人」",
 				"bodyList": [
 					{
 						"name": "tati_120113.webp",
@@ -9957,18 +6581,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120122.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "27",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「总而言之,今天帮大忙了」",
 				"bodyList": [
 					{
 						"name": "tati_120122.webp",
@@ -9979,18 +6599,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120107.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "28",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「谢礼嘛······」",
 				"bodyList": [
 					{
 						"name": "tati_120107.webp",
@@ -10001,18 +6617,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "29",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「就记在账单上吧」",
 				"bodyList": [
 					{
 						"name": "tati_120107.webp",
@@ -10023,18 +6635,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120112.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "30",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「了解—盛谢惠顾了哦?」",
 				"bodyList": [
 					{
 						"name": "tati_120112.webp",
@@ -10045,18 +6653,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "31",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "喀啷喀啷",
 				"bodyList": [
 					{
 						"name": "",
@@ -10067,18 +6671,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "32",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "门铃响起",
 				"bodyList": [
 					{
 						"name": "",
@@ -10089,18 +6689,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "33",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "喧哗瞬间安静下来。",
 				"bodyList": [
 					{
 						"name": "",
@@ -10111,18 +6707,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "34",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "进来的人是吉克。",
 				"bodyList": [
 					{
 						"name": "tati_340101.webp",
@@ -10133,18 +6725,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "35",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "是掌控着牢狱的组织之一,《不蚀金锁》的头目。",
 				"bodyList": [
 					{
 						"name": "tati_340101.webp",
@@ -10155,18 +6743,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "36",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "不仅组织的成员,就连店内一般的客人也对他以注目礼表示敬意。",
 				"bodyList": [
 					{
 						"name": "tati_340101.webp",
@@ -10177,18 +6761,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340102.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "37",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「各位继续吧」",
 				"bodyList": [
 					{
 						"name": "tati_340102.webp",
@@ -10199,18 +6779,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "38",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "仿佛停滞的时钟重新转动了一般,店内恢复了热闹的气氛。",
 				"bodyList": [
 					{
 						"name": "tati_340102.webp",
@@ -10221,18 +6797,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340101.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "39",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「抱歉,今天拜托你去做了无聊的工作」",
 				"bodyList": [
 					{
 						"name": "tati_340101.webp",
@@ -10243,18 +6815,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "40",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「不用介意」",
 				"bodyList": [
 					{
 						"name": "tati_340101.webp",
@@ -10265,18 +6833,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "41",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "吉克轻轻点了点头,在我右边坐了下来",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10287,18 +6851,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "42",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「逃跑的男人怎么样了?」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10309,18 +6869,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340104.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "43",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「嗯?已经不在这个世上了」",
 				"bodyList": [
 					{
 						"name": "tati_z340104.webp",
@@ -10331,18 +6887,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340107.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "44",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「有什么想要知道的事吗?」",
 				"bodyList": [
 					{
 						"name": "tati_z340107.webp",
@@ -10353,18 +6905,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "45",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「不,没什么」",
 				"bodyList": [
 					{
 						"name": "tati_z340107.webp",
@@ -10375,18 +6923,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340108.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "46",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「那个无聊的家伙,完全没有趣味呢」",
 				"bodyList": [
 					{
 						"name": "tati_z340108.webp",
@@ -10397,18 +6941,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340114.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "47",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「真希望他也替我负责清扫的部下也考虑考虑」",
 				"bodyList": [
 					{
 						"name": "tati_z340114.webp",
@@ -10419,18 +6959,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "48",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「真是灾难啊」",
 				"bodyList": [
 					{
 						"name": "tati_z340114.webp",
@@ -10441,18 +6977,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340102.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "49",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「比起那个,我听说了哦。你去追羽化病人了啊」",
 				"bodyList": [
 					{
 						"name": "tati_z340102.webp",
@@ -10463,18 +6995,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "50",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「消息真灵通」",
 				"bodyList": [
 					{
 						"name": "tati_z340102.webp",
@@ -10485,18 +7013,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340104.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "51",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「梅尔特也注意点」",
 				"bodyList": [
 					{
 						"name": "tati_z340104.webp",
@@ -10507,18 +7031,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340107.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "52",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「你丢钱已经不是一回两回了」",
 				"bodyList": [
 					{
 						"name": "tati_z340107.webp",
@@ -10529,18 +7049,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120132.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "53",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「好的—我会注意的。」",
 				"bodyList": [
 					{
 						"name": "tati_z340104.webp",
@@ -10556,18 +7072,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120101.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "54",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「吉克还是平常的点单吧」",
 				"bodyList": [
 					{
 						"name": "tati_z340102.webp",
@@ -10583,18 +7095,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120121.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "55",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「凯伊姆要再来一杯吗?」",
 				"bodyList": [
 					{
 						"name": "tati_z340102.webp",
@@ -10610,18 +7118,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "56",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "我们用眼神点头示意后,梅尔特开始准备起酒来。",
 				"bodyList": [
 					{
 						"name": "tati_z340102.webp",
@@ -10632,18 +7136,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "",
+				"index": "57",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "悠然地吐出眼圈后,吉克取出一个纸包放在柜台上。",
 				"bodyList": [
 					{
 						"name": "tati_z340102.webp",
@@ -10654,18 +7154,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340101.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "58",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「这是抓捕逃跑男人的报酬」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10676,18 +7172,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "59",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「下次有什么事再告诉我」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10698,18 +7190,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120102.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "60",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「来,久等了」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10725,18 +7213,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "61",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「话说回来梅尔特,为什么会被那种孩子偷到钱?」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10752,18 +7236,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340102.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "62",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「让我猜猜看」",
 				"bodyList": [
 					{
 						"name": "tati_z340102.webp",
@@ -10779,18 +7259,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340110.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "63",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「是那个吧,看某个特立独行的男人入迷了,所以就有了空隙?」",
 				"bodyList": [
 					{
 						"name": "tati_z340110.webp",
@@ -10806,18 +7282,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120113.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "64",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「可惜—」",
 				"bodyList": [
 					{
 						"name": "tati_z340110.webp",
@@ -10833,18 +7305,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120101.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "65",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「事实恰恰相反,是那家伙一直在纠缠我」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10860,18 +7328,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "66",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「完全把你当成新进的女佣了么」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10887,18 +7351,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120121.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "67",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「我从前可是很有名的,不会被当成这种下人吧」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10914,18 +7374,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120132.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "68",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「······而且,我没法对对我这么钟情的人发火啊」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -10941,18 +7397,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340106.webp",
 					"px": -300
 				},
-				"name": "凯伊姆&吉克",
+				"index": "69",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「你傻啊」",
 				"bodyList": [
 					{
 						"name": "tati_z340106.webp",
@@ -10968,18 +7420,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340112.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "70",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「异口同声呢,不亏是兄弟」",
 				"bodyList": [
 					{
 						"name": "tati_z340106.webp",
@@ -10995,18 +7443,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "71",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「别用这种称呼,怪恶心的」",
 				"bodyList": [
 					{
 						"name": "tati_z340106.webp",
@@ -11022,18 +7466,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340108.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "72",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「说得没错」",
 				"bodyList": [
 					{
 						"name": "tati_z340108.webp",
@@ -11049,18 +7489,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340105.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "73",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「······说起来······」",
 				"bodyList": [
 					{
 						"name": "tati_z340105.webp",
@@ -11076,18 +7512,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120107.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "74",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「怎么了?」",
 				"bodyList": [
 					{
 						"name": "tati_z340105.webp",
@@ -11103,18 +7535,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340107.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "75",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「有件事我一直很在意,我和凯伊姆,哪个是哥哥啊?」",
 				"bodyList": [
 					{
 						"name": "tati_z340107.webp",
@@ -11130,18 +7558,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "",
 					"px": -300
 				},
-				"name": "凯伊姆",
+				"index": "76",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「你也说这么无聊的话题」",
 				"bodyList": [
 					{
 						"name": "tati_z340107.webp",
@@ -11157,18 +7581,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340103.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "77",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「不,这是很重要的事情」",
 				"bodyList": [
 					{
 						"name": "tati_z340103.webp",
@@ -11184,18 +7604,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340101.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "78",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「梅尔特,事实是怎么样的?」",
 				"bodyList": [
 					{
 						"name": "tati_z340101.webp",
@@ -11211,18 +7627,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120113.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "79",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「啊~是怎么样的呢~」",
 				"bodyList": [
 					{
 						"name": "tati_z340105.webp",
@@ -11238,18 +7650,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_120112.webp",
 					"px": -300
 				},
-				"name": "梅尔特",
+				"index": "80",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「我忘记了」",
 				"bodyList": [
 					{
 						"name": "tati_z340105.webp",
@@ -11265,18 +7673,14 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 			},
 			{
 				"type": "cgtext",
-				"bg": "bg_3551.webp",
-				"memory": false,
 				"WindowSkin": false,
 				"head": {
 					"name": "face_340107.webp",
 					"px": -300
 				},
-				"name": "吉克",
+				"index": "81",
 				"time": 0,
 				"wait": 2000,
-				"sound": "",
-				"text": "「骗人」",
 				"bodyList": [
 					{
 						"name": "tati_z340107.webp",
@@ -11289,6 +7693,15 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 						"filter": false
 					}
 				]
+			},
+			{
+				"type": "changebg",
+				"img1": "bg_3551.webp",
+				"memory1": false,
+				"img2": "",
+				"memory2": false,
+				"time": 30,
+				"style": "引出"
 			}
 		],
 		"chapter05": null,
@@ -11323,6 +7736,138 @@ var events_c12a15a8_c380_4b28_8144_256cba95f760 =
 					}
 				]
 			}
+		],
+		"chapter06": null,
+		"chapter07": null,
+		"chapter000001": [
+			{
+				"type": "changebg",
+				"img1": "",
+				"memory1": false,
+				"img2": "",
+				"memory2": false,
+				"time": 30,
+				"style": "场景切换"
+			},
+			{
+				"type": "cgtextList",
+				"textList": "chapter001"
+			},
+			{
+				"type": "cgtext",
+				"bg": "",
+				"memory": false,
+				"WindowSkin": false,
+				"head": {
+					"name": "",
+					"px": -300
+				},
+				"index": "0",
+				"time": 30,
+				"wait": 1000,
+				"bodyList": [
+					{
+						"name": "",
+						"px": 100,
+						"filter": false
+					}
+				]
+			},
+			{
+				"type": "cgtext",
+				"bg": "",
+				"memory": false,
+				"WindowSkin": false,
+				"head": {
+					"name": "",
+					"px": -300
+				},
+				"index": "1",
+				"time": 30,
+				"wait": 1000,
+				"bodyList": [
+					{
+						"name": "",
+						"px": 100,
+						"filter": false
+					}
+				]
+			},
+			{
+				"type": "cgtext",
+				"bg": "",
+				"memory": false,
+				"WindowSkin": false,
+				"head": {
+					"name": "",
+					"px": -300
+				},
+				"index": "2",
+				"time": 30,
+				"wait": 1000,
+				"bodyList": [
+					{
+						"name": "",
+						"px": 100,
+						"filter": false
+					}
+				]
+			},
+			{
+				"type": "cgtext",
+				"bg": "",
+				"memory": false,
+				"WindowSkin": false,
+				"head": {
+					"name": "",
+					"px": -300
+				},
+				"index": "3",
+				"time": 30,
+				"wait": 1000,
+				"bodyList": [
+					{
+						"name": "",
+						"px": 100,
+						"filter": false
+					}
+				]
+			}
+		],
+		"chapter00002": [
+			{
+				"type": "changebg",
+				"img1": "",
+				"memory1": false,
+				"img2": "bg_3531.webp",
+				"memory2": false,
+				"time": 30,
+				"style": "引入"
+			},
+			{
+				"type": "cgtextList",
+				"textList": "chapter002"
+			},
+			{
+				"type": "cgtext",
+				"bg": "bg_3531.webp",
+				"memory": false,
+				"WindowSkin": false,
+				"head": {
+					"name": "",
+					"px": -300
+				},
+				"index": "0",
+				"time": 0,
+				"wait": 2000,
+				"bodyList": [
+					{
+						"name": "",
+						"px": 0,
+						"filter": false
+					}
+				]
+			}
 		]
 	},
 	"CommonEventTemplate": {
diff --git a/project/floors/Caimhome.js b/project/floors/Caimhome.js
index f93c9a3..da785fe 100644
--- a/project/floors/Caimhome.js
+++ b/project/floors/Caimhome.js
@@ -16,7 +16,7 @@ main.floors.Caimhome=
     "eachArrive": [],
     "parallelDo": "",
     "events": {
-        "3,7": [
+        "3,8": [
             {
                 "type": "setHeroOpacity",
                 "opacity": 0,
@@ -38,1213 +38,11 @@ main.floors.Caimhome=
                 "type": "waitAsync"
             },
             {
-                "type": "showBgFgMap",
-                "name": "fg2",
+                "type": "changeFloor",
+                "floorId": "zhujuejia",
                 "loc": [
-                    [
-                        0,
-                        0
-                    ],
-                    [
-                        0,
-                        1
-                    ],
-                    [
-                        0,
-                        2
-                    ],
-                    [
-                        0,
-                        3
-                    ],
-                    [
-                        0,
-                        4
-                    ],
-                    [
-                        0,
-                        5
-                    ],
-                    [
-                        0,
-                        6
-                    ],
-                    [
-                        0,
-                        7
-                    ],
-                    [
-                        0,
-                        8
-                    ],
-                    [
-                        1,
-                        0
-                    ],
-                    [
-                        1,
-                        1
-                    ],
-                    [
-                        1,
-                        2
-                    ],
-                    [
-                        1,
-                        3
-                    ],
-                    [
-                        1,
-                        4
-                    ],
-                    [
-                        1,
-                        5
-                    ],
-                    [
-                        1,
-                        6
-                    ],
-                    [
-                        1,
-                        7
-                    ],
-                    [
-                        1,
-                        8
-                    ],
-                    [
-                        2,
-                        0
-                    ],
-                    [
-                        2,
-                        1
-                    ],
-                    [
-                        2,
-                        2
-                    ],
-                    [
-                        2,
-                        3
-                    ],
-                    [
-                        2,
-                        4
-                    ],
-                    [
-                        2,
-                        5
-                    ],
-                    [
-                        2,
-                        6
-                    ],
-                    [
-                        2,
-                        7
-                    ],
-                    [
-                        2,
-                        8
-                    ],
-                    [
-                        3,
-                        0
-                    ],
-                    [
-                        3,
-                        1
-                    ],
-                    [
-                        3,
-                        2
-                    ],
-                    [
-                        3,
-                        3
-                    ],
-                    [
-                        3,
-                        4
-                    ],
-                    [
-                        3,
-                        5
-                    ],
-                    [
-                        3,
-                        6
-                    ],
-                    [
-                        3,
-                        7
-                    ],
-                    [
-                        3,
-                        8
-                    ],
-                    [
-                        4,
-                        0
-                    ],
-                    [
-                        4,
-                        1
-                    ],
-                    [
-                        4,
-                        2
-                    ],
-                    [
-                        4,
-                        3
-                    ],
-                    [
-                        4,
-                        4
-                    ],
-                    [
-                        4,
-                        5
-                    ],
-                    [
-                        4,
-                        6
-                    ],
-                    [
-                        4,
-                        7
-                    ],
-                    [
-                        4,
-                        8
-                    ],
-                    [
-                        5,
-                        0
-                    ],
-                    [
-                        5,
-                        1
-                    ],
-                    [
-                        5,
-                        2
-                    ],
-                    [
-                        5,
-                        3
-                    ],
-                    [
-                        5,
-                        4
-                    ],
-                    [
-                        5,
-                        5
-                    ],
-                    [
-                        5,
-                        6
-                    ],
-                    [
-                        5,
-                        7
-                    ],
-                    [
-                        5,
-                        8
-                    ],
-                    [
-                        6,
-                        0
-                    ],
-                    [
-                        6,
-                        1
-                    ],
-                    [
-                        6,
-                        2
-                    ],
-                    [
-                        6,
-                        3
-                    ],
-                    [
-                        6,
-                        4
-                    ],
-                    [
-                        6,
-                        5
-                    ],
-                    [
-                        6,
-                        6
-                    ],
-                    [
-                        6,
-                        7
-                    ],
-                    [
-                        6,
-                        8
-                    ],
-                    [
-                        7,
-                        0
-                    ],
-                    [
-                        7,
-                        1
-                    ],
-                    [
-                        7,
-                        2
-                    ],
-                    [
-                        7,
-                        3
-                    ],
-                    [
-                        7,
-                        4
-                    ],
-                    [
-                        7,
-                        5
-                    ],
-                    [
-                        7,
-                        6
-                    ],
-                    [
-                        7,
-                        7
-                    ],
-                    [
-                        7,
-                        8
-                    ],
-                    [
-                        8,
-                        0
-                    ],
-                    [
-                        8,
-                        1
-                    ],
-                    [
-                        8,
-                        2
-                    ],
-                    [
-                        8,
-                        3
-                    ],
-                    [
-                        8,
-                        4
-                    ],
-                    [
-                        8,
-                        5
-                    ],
-                    [
-                        8,
-                        6
-                    ],
-                    [
-                        8,
-                        7
-                    ],
-                    [
-                        8,
-                        8
-                    ],
-                    [
-                        9,
-                        0
-                    ],
-                    [
-                        9,
-                        1
-                    ],
-                    [
-                        9,
-                        2
-                    ],
-                    [
-                        9,
-                        3
-                    ],
-                    [
-                        9,
-                        4
-                    ],
-                    [
-                        9,
-                        5
-                    ],
-                    [
-                        9,
-                        6
-                    ],
-                    [
-                        9,
-                        7
-                    ],
-                    [
-                        9,
-                        8
-                    ],
-                    [
-                        10,
-                        0
-                    ],
-                    [
-                        10,
-                        1
-                    ],
-                    [
-                        10,
-                        2
-                    ],
-                    [
-                        10,
-                        3
-                    ],
-                    [
-                        10,
-                        4
-                    ],
-                    [
-                        10,
-                        5
-                    ],
-                    [
-                        10,
-                        6
-                    ],
-                    [
-                        10,
-                        7
-                    ],
-                    [
-                        10,
-                        8
-                    ],
-                    [
-                        11,
-                        0
-                    ],
-                    [
-                        11,
-                        1
-                    ],
-                    [
-                        11,
-                        2
-                    ],
-                    [
-                        11,
-                        3
-                    ],
-                    [
-                        11,
-                        4
-                    ],
-                    [
-                        11,
-                        5
-                    ],
-                    [
-                        11,
-                        6
-                    ],
-                    [
-                        11,
-                        7
-                    ],
-                    [
-                        11,
-                        8
-                    ],
-                    [
-                        12,
-                        0
-                    ],
-                    [
-                        12,
-                        1
-                    ],
-                    [
-                        12,
-                        2
-                    ],
-                    [
-                        12,
-                        3
-                    ],
-                    [
-                        12,
-                        4
-                    ],
-                    [
-                        12,
-                        5
-                    ],
-                    [
-                        12,
-                        6
-                    ],
-                    [
-                        12,
-                        7
-                    ],
-                    [
-                        12,
-                        8
-                    ],
-                    [
-                        13,
-                        0
-                    ],
-                    [
-                        13,
-                        1
-                    ],
-                    [
-                        13,
-                        2
-                    ],
-                    [
-                        13,
-                        3
-                    ],
-                    [
-                        13,
-                        4
-                    ],
-                    [
-                        13,
-                        5
-                    ],
-                    [
-                        13,
-                        6
-                    ],
-                    [
-                        13,
-                        7
-                    ],
-                    [
-                        13,
-                        8
-                    ],
-                    [
-                        7,
-                        9
-                    ],
-                    [
-                        7,
-                        10
-                    ],
-                    [
-                        7,
-                        11
-                    ],
-                    [
-                        8,
-                        9
-                    ],
-                    [
-                        8,
-                        10
-                    ],
-                    [
-                        8,
-                        11
-                    ],
-                    [
-                        9,
-                        9
-                    ],
-                    [
-                        9,
-                        10
-                    ],
-                    [
-                        9,
-                        11
-                    ],
-                    [
-                        10,
-                        9
-                    ],
-                    [
-                        10,
-                        10
-                    ],
-                    [
-                        10,
-                        11
-                    ],
-                    [
-                        11,
-                        9
-                    ],
-                    [
-                        11,
-                        10
-                    ],
-                    [
-                        11,
-                        11
-                    ]
-                ]
-            },
-            {
-                "type": "setBlock",
-                "number": "starPortal",
-                "loc": [
-                    [
-                        3,
-                        10
-                    ]
-                ]
-            },
-            {
-                "type": "moveHero",
-                "steps": [
-                    "down:4"
-                ]
-            },
-            {
-                "type": "setCurtain",
-                "color": [
-                    0,
-                    0,
-                    0,
-                    0
-                ],
-                "time": 500,
-                "keep": true
-            },
-            {
-                "type": "setHeroOpacity",
-                "opacity": 1,
-                "time": 500
-            }
-        ],
-        "3,10": [
-            {
-                "type": "setHeroOpacity",
-                "opacity": 0,
-                "time": 500
-            },
-            {
-                "type": "setCurtain",
-                "color": [
-                    0,
-                    0,
-                    0,
-                    1
-                ],
-                "time": 500,
-                "keep": true,
-                "async": true
-            },
-            {
-                "type": "waitAsync"
-            },
-            {
-                "type": "hideBgFgMap",
-                "name": "fg2",
-                "loc": [
-                    [
-                        0,
-                        0
-                    ],
-                    [
-                        0,
-                        1
-                    ],
-                    [
-                        0,
-                        2
-                    ],
-                    [
-                        0,
-                        3
-                    ],
-                    [
-                        0,
-                        4
-                    ],
-                    [
-                        0,
-                        5
-                    ],
-                    [
-                        0,
-                        6
-                    ],
-                    [
-                        0,
-                        7
-                    ],
-                    [
-                        0,
-                        8
-                    ],
-                    [
-                        1,
-                        0
-                    ],
-                    [
-                        1,
-                        1
-                    ],
-                    [
-                        1,
-                        2
-                    ],
-                    [
-                        1,
-                        3
-                    ],
-                    [
-                        1,
-                        4
-                    ],
-                    [
-                        1,
-                        5
-                    ],
-                    [
-                        1,
-                        6
-                    ],
-                    [
-                        1,
-                        7
-                    ],
-                    [
-                        1,
-                        8
-                    ],
-                    [
-                        2,
-                        0
-                    ],
-                    [
-                        2,
-                        1
-                    ],
-                    [
-                        2,
-                        2
-                    ],
-                    [
-                        2,
-                        3
-                    ],
-                    [
-                        2,
-                        4
-                    ],
-                    [
-                        2,
-                        5
-                    ],
-                    [
-                        2,
-                        6
-                    ],
-                    [
-                        2,
-                        7
-                    ],
-                    [
-                        2,
-                        8
-                    ],
-                    [
-                        3,
-                        0
-                    ],
-                    [
-                        3,
-                        1
-                    ],
-                    [
-                        3,
-                        2
-                    ],
-                    [
-                        3,
-                        3
-                    ],
-                    [
-                        3,
-                        4
-                    ],
-                    [
-                        3,
-                        5
-                    ],
-                    [
-                        3,
-                        6
-                    ],
-                    [
-                        3,
-                        7
-                    ],
-                    [
-                        3,
-                        8
-                    ],
-                    [
-                        4,
-                        0
-                    ],
-                    [
-                        4,
-                        1
-                    ],
-                    [
-                        4,
-                        2
-                    ],
-                    [
-                        4,
-                        3
-                    ],
-                    [
-                        4,
-                        4
-                    ],
-                    [
-                        4,
-                        5
-                    ],
-                    [
-                        4,
-                        6
-                    ],
-                    [
-                        4,
-                        7
-                    ],
-                    [
-                        4,
-                        8
-                    ],
-                    [
-                        5,
-                        0
-                    ],
-                    [
-                        5,
-                        1
-                    ],
-                    [
-                        5,
-                        2
-                    ],
-                    [
-                        5,
-                        3
-                    ],
-                    [
-                        5,
-                        4
-                    ],
-                    [
-                        5,
-                        5
-                    ],
-                    [
-                        5,
-                        6
-                    ],
-                    [
-                        5,
-                        7
-                    ],
-                    [
-                        5,
-                        8
-                    ],
-                    [
-                        6,
-                        0
-                    ],
-                    [
-                        6,
-                        1
-                    ],
-                    [
-                        6,
-                        2
-                    ],
-                    [
-                        6,
-                        3
-                    ],
-                    [
-                        6,
-                        4
-                    ],
-                    [
-                        6,
-                        5
-                    ],
-                    [
-                        6,
-                        6
-                    ],
-                    [
-                        6,
-                        7
-                    ],
-                    [
-                        6,
-                        8
-                    ],
-                    [
-                        7,
-                        0
-                    ],
-                    [
-                        7,
-                        1
-                    ],
-                    [
-                        7,
-                        2
-                    ],
-                    [
-                        7,
-                        3
-                    ],
-                    [
-                        7,
-                        4
-                    ],
-                    [
-                        7,
-                        5
-                    ],
-                    [
-                        7,
-                        6
-                    ],
-                    [
-                        7,
-                        7
-                    ],
-                    [
-                        7,
-                        8
-                    ],
-                    [
-                        8,
-                        0
-                    ],
-                    [
-                        8,
-                        1
-                    ],
-                    [
-                        8,
-                        2
-                    ],
-                    [
-                        8,
-                        3
-                    ],
-                    [
-                        8,
-                        4
-                    ],
-                    [
-                        8,
-                        5
-                    ],
-                    [
-                        8,
-                        6
-                    ],
-                    [
-                        8,
-                        7
-                    ],
-                    [
-                        8,
-                        8
-                    ],
-                    [
-                        9,
-                        0
-                    ],
-                    [
-                        9,
-                        1
-                    ],
-                    [
-                        9,
-                        2
-                    ],
-                    [
-                        9,
-                        3
-                    ],
-                    [
-                        9,
-                        4
-                    ],
-                    [
-                        9,
-                        5
-                    ],
-                    [
-                        9,
-                        6
-                    ],
-                    [
-                        9,
-                        7
-                    ],
-                    [
-                        9,
-                        8
-                    ],
-                    [
-                        10,
-                        0
-                    ],
-                    [
-                        10,
-                        1
-                    ],
-                    [
-                        10,
-                        2
-                    ],
-                    [
-                        10,
-                        3
-                    ],
-                    [
-                        10,
-                        4
-                    ],
-                    [
-                        10,
-                        5
-                    ],
-                    [
-                        10,
-                        6
-                    ],
-                    [
-                        10,
-                        7
-                    ],
-                    [
-                        10,
-                        8
-                    ],
-                    [
-                        11,
-                        0
-                    ],
-                    [
-                        11,
-                        1
-                    ],
-                    [
-                        11,
-                        2
-                    ],
-                    [
-                        11,
-                        3
-                    ],
-                    [
-                        11,
-                        4
-                    ],
-                    [
-                        11,
-                        5
-                    ],
-                    [
-                        11,
-                        6
-                    ],
-                    [
-                        11,
-                        7
-                    ],
-                    [
-                        11,
-                        8
-                    ],
-                    [
-                        12,
-                        0
-                    ],
-                    [
-                        12,
-                        1
-                    ],
-                    [
-                        12,
-                        2
-                    ],
-                    [
-                        12,
-                        3
-                    ],
-                    [
-                        12,
-                        4
-                    ],
-                    [
-                        12,
-                        5
-                    ],
-                    [
-                        12,
-                        6
-                    ],
-                    [
-                        12,
-                        7
-                    ],
-                    [
-                        12,
-                        8
-                    ],
-                    [
-                        13,
-                        0
-                    ],
-                    [
-                        13,
-                        1
-                    ],
-                    [
-                        13,
-                        2
-                    ],
-                    [
-                        13,
-                        3
-                    ],
-                    [
-                        13,
-                        4
-                    ],
-                    [
-                        13,
-                        5
-                    ],
-                    [
-                        13,
-                        6
-                    ],
-                    [
-                        13,
-                        7
-                    ],
-                    [
-                        13,
-                        8
-                    ],
-                    [
-                        7,
-                        9
-                    ],
-                    [
-                        7,
-                        10
-                    ],
-                    [
-                        7,
-                        11
-                    ],
-                    [
-                        8,
-                        9
-                    ],
-                    [
-                        8,
-                        10
-                    ],
-                    [
-                        8,
-                        11
-                    ],
-                    [
-                        9,
-                        9
-                    ],
-                    [
-                        9,
-                        10
-                    ],
-                    [
-                        9,
-                        11
-                    ],
-                    [
-                        10,
-                        9
-                    ],
-                    [
-                        10,
-                        10
-                    ],
-                    [
-                        10,
-                        11
-                    ],
-                    [
-                        11,
-                        9
-                    ],
-                    [
-                        11,
-                        10
-                    ],
-                    [
-                        11,
-                        11
-                    ]
-                ]
-            },
-            {
-                "type": "setBlock",
-                "number": "X50030",
-                "loc": [
-                    [
-                        3,
-                        10
-                    ]
-                ]
-            },
-            {
-                "type": "moveHero",
-                "steps": [
-                    "up:3"
+                    5,
+                    2
                 ]
             },
             {
@@ -1286,13 +84,13 @@ main.floors.Caimhome=
     [ 17,10173,20130,40091,40099,40155, 17,  0,71224,90520,  0,40172, 17],
     [ 17,  0,  0,71182,71183,  0,  0,  0,  0,  0,  0,  0, 17],
     [ 17,71681,  0,  0,  0,  0, 17, 17,  0,  0, 17, 17, 17],
-    [ 17,71689,  0,  0,  0,  0, 17, 17,71606,71631,60111,71678, 17],
+    [ 17,71689,  0,  0,  0,  0,140, 17,71606,71631,60111,71678, 17],
     [ 17,  0,  0,  0,  0,  0, 17,71880, 17,71639,71686,71603, 17],
-    [ 17, 17, 17, 17, 17,  0,  0,  0,  0,  0,  0,  0, 17],
-    [120165,120166,120166,120166,120166,120166,120167, 17,  0,  0,  0, 17,120165],
-    [120173,120174,120174, 90,90212,120174,120175, 17,  0,  0,71336, 17,120173],
-    [  0, 17,80007,  0,120073,80011,40110, 17, 17, 17, 17, 17,  0],
-    [  0, 17,80003,  0,  0,  0,80010,120165,120166,120166,120166,120167,40118]
+    [ 17,  0,  0,  0,  0,  0, 17,  0,  0,  0,  0,  0, 17],
+    [  0, 17, 17, 17, 17,  0,  0,  0,  0,  0,  0, 17,120165],
+    [120165,120166,120172,200550,120168,120166,120167, 17,  0,  0,71336, 17,120173],
+    [120173,  0,120180,200558,120176,90214,120175, 17, 17, 17, 17, 17,  0],
+    [  0,120073,  0,  0,  0,  0,  0,120165,120166,120166,120166,120167,40118]
 ],
     "areas": "下层",
     "bgmap": [
@@ -1303,12 +101,12 @@ main.floors.Caimhome=
     [  0,70071,70070,70070,70070,70070,  0,70070,70070,70071,70070,70062,  0],
     [  0,70070,70062,70070,70070,70071,  0,  0,  0,  0,  0,  0,  0],
     [  0,70070,70070,70063,70070,70070,40017,40017,40017,40017,40017,40017,  0],
-    [  0,70071,70062,70070,70070,70062,40025,40025,40025,40025,40025,40025,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,70071,70062,70070,  0,  0],
+    [  0,70071,70062,70070,70070,70062,40017,40025,40025,40025,40025,40025,  0],
+    [  0,70062,70071,70071,70062,70071,40025,70062,70071,70062,70070,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,70070,70070,70062,  0,  0],
-    [  0,  0,  0,50030,120174,  0,  0,  0,70062,70070,70063,  0,  0],
-    [  0,  0,  0,10002,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,10010,  0,308,  0,  0,  0,  0,  0,  0,  0]
+    [90078,90078,90078,90078,90078,90078,90078,  0,70062,70070,70063,  0,  0],
+    [90078,120174,90078,120174,90078,120174,90078,  0,  0,  0,  0,  0,90078],
+    [90078,90078,90078,90078,90078,90078,90078,  0,  0,  0,  0,  0,90078]
 ],
     "fgmap": [
     [140,140,140,140,140,140,140,  0,  0,  0,  0,  0,140],
@@ -1317,13 +115,13 @@ main.floors.Caimhome=
     [140,  0,  0,  0,  0,  0,  0,  0,  0,10141,  0,  0,140],
     [140,71673,  0,  0,  0,  0,140,10165,  0,  0,70624,70625,140],
     [140,71586,  0,  0,  0,  0,140,140,140,140,140,140,140],
+    [140,  0,  0,  0,  0,  0,140,  0,  0,  0,  0,  0,140],
     [140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [140,70630,70631,  0,50062,  0,  0,  0,  0,  0,  0,  0,140],
-    [140,140,140,140,140,140,140,140,  0,  0,  0,140,140],
-    [  0,90207,90206,  0,90204,90206,  0,140,  0,  0,71320,140,110072],
-    [  0,90215,90214,  0,120065,90214,40102,140,30114,344,20059,140,110080],
-    [  0,  0,  0,  0,  0,  0,80013,140,140,140,140,140,110088],
-    [  0,  0,  0,  0,  0,  0,  0,  0,120072,  0,  0,  0,  0]
+    [140,70630,70631,  0,50062,  0,  0,  0,  0,  0,  0,140,140],
+    [140,140,140,140,140,140,140,140,  0,  0,71320,140,110072],
+    [  0,  0,  0,  0,140091,140092,140092,140,30114,344,20059,140,110080],
+    [120039,  0,  0,  0,140099,140100,140100,140,140,140,140,140,110088],
+    [120017,  0,  0,  0,140107,140108,140108,  0,120072,90223,  0,  0,80013]
 ],
     "bg2map": [
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
@@ -1333,26 +131,14 @@ main.floors.Caimhome=
     [  0,  0,  0,30150,30150,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0, 90,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,71694,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0, 90,  0,  0,  0,  0,  0,  0,71694,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,80020,  0,  0,  0,  0,  0,  0]
+    [  0,120065,  0,  0,  0,120174,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,90244,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "fg2map": [
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [80041,80042,80043,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [80049,80050,80051,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [80057,80058,80059,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [80065,80066,80067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+
 ]
 }
\ No newline at end of file
diff --git a/project/floors/KTV.js b/project/floors/KTV.js
index eceffe1..5fd3f57 100644
--- a/project/floors/KTV.js
+++ b/project/floors/KTV.js
@@ -11,11 +11,201 @@ main.floors.KTV=
     "images": [],
     "ratio": 1,
     "defaultGround": "grass",
-    "firstArrive": [],
+    "firstArrive": [
+        {
+            "type": "changebg",
+            "img1": "",
+            "memory1": false,
+            "img2": "bg_3531.webp",
+            "memory2": false,
+            "time": 30,
+            "style": "引入"
+        },
+        {
+            "type": "cgtextList",
+            "textList": "chapter002"
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "0",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "1",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "2",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "3",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "4",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "5",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "6",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "cgtext",
+            "WindowSkin": false,
+            "head": {
+                "name": "",
+                "px": -300
+            },
+            "index": "7",
+            "time": 0,
+            "wait": 2000,
+            "bodyList": [
+                {
+                    "name": "",
+                    "px": 0,
+                    "filter": false
+                }
+            ]
+        },
+        {
+            "type": "changebg",
+            "img1": "bg_3531.webp",
+            "memory1": false,
+            "img2": "",
+            "memory2": false,
+            "time": 30,
+            "style": "引出"
+        },
+        {
+            "type": "setHeroOpacity",
+            "opacity": 1
+        },
+        {
+            "type": "setCurtain",
+            "time": 1000
+        }
+    ],
     "eachArrive": [],
     "parallelDo": "",
     "events": {},
-    "changeFloor": {},
+    "changeFloor": {
+        "12,10": {
+            "floorId": "changguan2",
+            "loc": [
+                2,
+                1
+            ]
+        },
+        "10,1": {
+            "floorId": "KTVF2",
+            "loc": [
+                10,
+                4
+            ]
+        }
+    },
     "beforeBattle": {},
     "afterBattle": {},
     "afterGetItem": {},
@@ -32,7 +222,7 @@ main.floors.KTV=
     [ 17,71797,71797,71797,71797,71797,71797,71797,71797,71797, 90, 17,  0],
     [ 17,90211,90211,90211,90211,71877,90211,90211,  0,90587,  0, 17,  0],
     [ 17,90541,90542,90543, 17,71761, 17, 17,71288,71116,  0, 17,  0],
-    [ 17,  0,30216,  0,  0,  0,  0,  0,  0,71257,  0, 17,  0],
+    [ 17,  0,30216,  0,  0,123,  0,  0,  0,71257,  0, 17,  0],
     [ 17,  0,  0,  0,71568,71754,20167,  0,  0,71265,  0, 17,  0],
     [ 17,  0,  0,  0,71887,  0,  0,  0,60160,71265,  0, 17,  0],
     [ 17,71340, 17, 17,  0,  0,  0, 17, 17,70568,  0,71796,71797],
diff --git a/project/floors/KTVF2.js b/project/floors/KTVF2.js
index 78260da..62626a0 100644
--- a/project/floors/KTVF2.js
+++ b/project/floors/KTVF2.js
@@ -15,7 +15,15 @@ main.floors.KTVF2=
     "eachArrive": [],
     "parallelDo": "",
     "events": {},
-    "changeFloor": {},
+    "changeFloor": {
+        "10,4": {
+            "floorId": "KTV",
+            "loc": [
+                10,
+                1
+            ]
+        }
+    },
     "beforeBattle": {},
     "afterBattle": {},
     "afterGetItem": {},
diff --git a/project/floors/changguan1.js b/project/floors/changguan1.js
new file mode 100644
index 0000000..833d90e
--- /dev/null
+++ b/project/floors/changguan1.js
@@ -0,0 +1,140 @@
+main.floors.changguan1=
+{
+    "floorId": "changguan1",
+    "title": "主塔 10 层",
+    "name": "10",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "grass2",
+    "firstArrive": [
+        {
+            "type": "setCurtain",
+            "time": 1000
+        },
+        {
+            "type": "autoText",
+            "text": "通称——",
+            "time": 2000
+        },
+        {
+            "type": "setCurtain",
+            "color": [
+                0,
+                0,
+                0,
+                1
+            ],
+            "time": 1000,
+            "keep": true
+        },
+        {
+            "type": "changeFloor",
+            "floorId": "changguan2",
+            "loc": [
+                0,
+                0
+            ]
+        }
+    ],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "6,0": {
+            "floorId": "guangchang",
+            "loc": [
+                6,
+                12
+            ]
+        },
+        "5,12": {
+            "floorId": "changguan2",
+            "loc": [
+                5,
+                0
+            ]
+        },
+        "6,12": {
+            "floorId": "changguan2",
+            "loc": [
+                6,
+                0
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [210178,210178,210178,210178,210171,  0, 91,  0,210171,110272,110273,110274,110275],
+    [110241,110242,110243,110244,210187,  0,  0,  0,210187,110280,110281,110282,110283],
+    [110249,110250,110251,110252,210195,  0,  0,  0,210195,110288,110289,110290,110291],
+    [110257,110258,110259,200199,  0,  0,  0,  0,  0,110296,110297,110298,110299],
+    [110265,110266,110267,110268,  0,  0,  0,  0,  0,110304,110305,110306,110307],
+    [110273,110274,110275,110276,  0,  0,  0,  0,  0,110116,110313,110314,110315],
+    [110281,110282,110283,110284,  0,  0,  0,  0,  0,110132,  0,  0,  0],
+    [110289,110290,110291,110292,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [110297,110298,110299,110300,  0,  0,  0,  0,  0,110116,  0,  0,  0],
+    [110305,110306,110307,110308,  0,  0,  0,  0,  0,110261,110262,110262,110263],
+    [110313,110314,110315,110316,110122,110122,  0,110122,110122,110269,110270,110270,110271],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,110277,110278,110278,110279],
+    [140,140186,120002,120003,  0, 93, 93,  0,  0,110285,110286,110286,110286]
+],
+    "bgmap": [
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
+],
+    "fgmap": [
+    [  0,  0,  0,200175,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,200183,  0,  0,  0,  0,140086,  0,  0,  0,  0],
+    [  0,  0,  0,200191,  0,  0,  0,  0,140094,  0,  0,  0,  0],
+    [  0,  0,  0,110260,  0,  0,  0,  0,140102,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,140110,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,110246,110246,110246,110246],
+    [140091,140092,140093,  0,110114,110114,  0,110114,110114,  0,  0,  0,  0],
+    [140099,140100,140101,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140107,140108,140109,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,130046,130046,130046,130046,130046,210178,  0,  0,210178],
+    [210186,210186,210186,210186,130046,130046,130046,130046,130046,  0,  0,  0,  0],
+    [210194,210194,210194,210194,130054,130054,130054,130054,130054,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,110312,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/changguan2.js b/project/floors/changguan2.js
new file mode 100644
index 0000000..c0e530d
--- /dev/null
+++ b/project/floors/changguan2.js
@@ -0,0 +1,148 @@
+main.floors.changguan2=
+{
+    "floorId": "changguan2",
+    "title": "街道",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X20003",
+    "firstArrive": [
+        {
+            "type": "setCurtain",
+            "time": 1000
+        },
+        {
+            "type": "autoText",
+            "text": "《牢狱》",
+            "time": 2000
+        },
+        {
+            "type": "setCurtain",
+            "color": [
+                0,
+                0,
+                0,
+                1
+            ],
+            "time": 1000,
+            "keep": true
+        },
+        {
+            "type": "changeFloor",
+            "floorId": "KTV",
+            "loc": [
+                10,
+                8
+            ]
+        }
+    ],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "5,0": {
+            "floorId": "changguan1",
+            "loc": [
+                5,
+                12
+            ]
+        },
+        "6,0": {
+            "floorId": "changguan1",
+            "loc": [
+                6,
+                12
+            ]
+        },
+        "12,8": {
+            "floorId": "street01",
+            "loc": [
+                0,
+                8
+            ]
+        },
+        "2,1": {
+            "floorId": "KTV",
+            "loc": [
+                12,
+                10
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [140193,130178,120010,120011,  0, 91, 91,  0,140184,140,140,140,140186],
+    [140193,130178, 90,  0,  0,  0,  0,  0,140192,140,140,140,140194],
+    [140193,130178,40163,120019,  0,  0,  0,  0,140200,140195,140,140,140194],
+    [140193,143,143,120027,  0,  0,  0,  0,70128,140192,140,140,140194],
+    [140193,210148,143,143,  0,  0,  0,71826,70136,140192,140,140,140194],
+    [140193,143,143,143,71226,  0,  0,  0,50151,140200,140201,140201,140202],
+    [70165,70165,70165,70166,71819,  0,  0,  0,  0,70128,70129,70129,70130],
+    [140193,210153,143,70174,71827,  0,  0,  0,71826,120056,70137,70137,70138],
+    [140193,210153,210156,143,71695,  0,  0,  0,  0,  0,  0,  0, 94],
+    [140193,143,143,143,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140193,140193,130178,70130,  0,  0,  0,  0,  0,  0,  0,140,140],
+    [140193,140193,130178,40165,200199,  0,  0,71791,  0,  0,140192,140,140],
+    [140193,140193,130186,110114,110114, 93, 93,  0,143,  0,140192,140,140]
+],
+    "areas": "牢狱",
+    "bgmap": [
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
+],
+    "fgmap": [
+    [140,140194,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140194,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140194,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140,140186,  0,  0,  0,71818,  0,  0,  0,  0,  0],
+    [140,140,140,140194,71218,  0,  0,  0,71818,  0,  0,  0,  0],
+    [140201,140201,140201,140202,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,71685,  0,  0,  0,71818,  0,90587,  0,  0],
+    [140,140,140,140186,71844,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140,140194,200175,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140196,140202,200183,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140194,  0,200191,  0,  0,  0,  0,71818,140184,140,140],
+    [140,140,140194,  0,  0,  0,  0,  0,71832,71833,140192,140,140],
+    [140,140,140194,  0,110117,  0,  0,140184,140,140,140,140,140]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,110293,110294,110294,110294,110294],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,71826,  0,  0,  0,  0],
+    [  0,  0,  0,  0,71693,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,70136,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,120064,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,70138,  0,  0,  0,  0,  0,71826,  0,  0,  0],
+    [  0,  0,  0,40173,  0,  0,  0,71799,71840,71841,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/chushi.js b/project/floors/chushi.js
new file mode 100644
index 0000000..2ca6465
--- /dev/null
+++ b/project/floors/chushi.js
@@ -0,0 +1,129 @@
+main.floors.chushi=
+{
+    "floorId": "chushi",
+    "title": "开场剧情",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "T400",
+    "firstArrive": [
+        {
+            "type": "insert",
+            "name": "chapter0"
+        },
+        {
+            "type": "choices",
+            "text": "要进行教程内容吗",
+            "choices": [
+                {
+                    "text": "确定",
+                    "action": [
+                        {
+                            "type": "changeFloor",
+                            "floorId": "jiaocheng01",
+                            "loc": [
+                                6,
+                                12
+                            ]
+                        }
+                    ]
+                },
+                {
+                    "text": "取消",
+                    "action": [
+                        {
+                            "type": "changeFloor",
+                            "floorId": "guangchang",
+                            "loc": [
+                                6,
+                                10
+                            ]
+                        }
+                    ]
+                },
+                {
+                    "text": "一区测试",
+                    "action": [
+                        {
+                            "type": "changeFloor",
+                            "floorId": "yiqu1",
+                            "loc": [
+                                0,
+                                7
+                            ]
+                        },
+                        {
+                            "type": "setValue",
+                            "name": "status:atk",
+                            "value": "16"
+                        },
+                        {
+                            "type": "setValue",
+                            "name": "status:def",
+                            "value": "16"
+                        },
+                        {
+                            "type": "setValue",
+                            "name": "status:spell",
+                            "value": "0"
+                        },
+                        {
+                            "type": "setValue",
+                            "name": "status:speed",
+                            "value": "20"
+                        },
+                        {
+                            "type": "setValue",
+                            "name": "status:hp",
+                            "value": "1000"
+                        }
+                    ]
+                }
+            ]
+        }
+    ],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {},
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "areas": "开场剧情",
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/guangchang.js b/project/floors/guangchang.js
index 2adfdaa..3427e91 100644
--- a/project/floors/guangchang.js
+++ b/project/floors/guangchang.js
@@ -10,27 +10,276 @@ main.floors.guangchang=
     "canUseQuickShop": true,
     "images": [],
     "ratio": 1,
-    "defaultGround": "grass2",
-    "firstArrive": [],
+    "defaultGround": "X90078",
+    "firstArrive": [
+        {
+            "type": "setCurtain",
+            "time": 100
+        },
+        {
+            "type": "setText",
+            "position": "center"
+        },
+        {
+            "type": "move",
+            "loc": [
+                6,
+                7
+            ],
+            "time": 500,
+            "keep": true,
+            "async": true,
+            "steps": [
+                "up:1",
+                "left:1",
+                "up:2"
+            ]
+        },
+        {
+            "type": "sleep",
+            "time": 400
+        },
+        {
+            "type": "move",
+            "loc": [
+                8,
+                8
+            ],
+            "time": 500,
+            "keep": true,
+            "async": true,
+            "steps": [
+                "up:1",
+                "left:2"
+            ]
+        },
+        {
+            "type": "sleep",
+            "time": 500
+        },
+        {
+            "type": "move",
+            "loc": [
+                4,
+                9
+            ],
+            "time": 500,
+            "keep": true,
+            "steps": [
+                "right:1"
+            ]
+        },
+        {
+            "type": "waitAsync"
+        },
+        {
+            "type": "autoText",
+            "text": "浮游都市,《诺瓦斯·艾蒂尔》。",
+            "time": 2000
+        },
+        {
+            "type": "move",
+            "loc": [
+                6,
+                5
+            ],
+            "time": 500,
+            "keep": true,
+            "async": true,
+            "steps": [
+                "up:2"
+            ]
+        },
+        {
+            "type": "move",
+            "loc": [
+                6,
+                7
+            ],
+            "time": 500,
+            "keep": true,
+            "steps": [
+                "up:1"
+            ]
+        },
+        {
+            "type": "waitAsync"
+        },
+        {
+            "type": "sleep",
+            "time": 2000
+        },
+        {
+            "type": "move",
+            "loc": [
+                6,
+                3
+            ],
+            "time": 500,
+            "async": true,
+            "steps": [
+                "down:9"
+            ]
+        },
+        {
+            "type": "move",
+            "loc": [
+                5,
+                4
+            ],
+            "time": 500,
+            "keep": true,
+            "async": true,
+            "steps": [
+                "up:1",
+                "right:1"
+            ]
+        },
+        {
+            "type": "move",
+            "loc": [
+                6,
+                6
+            ],
+            "time": 300,
+            "keep": true,
+            "async": true,
+            "steps": [
+                "right:2"
+            ]
+        },
+        {
+            "type": "sleep",
+            "time": 1000
+        },
+        {
+            "type": "move",
+            "loc": [
+                8,
+                6
+            ],
+            "time": 500,
+            "keep": true,
+            "steps": [
+                "left:1"
+            ]
+        },
+        {
+            "type": "sleep",
+            "time": 2000
+        },
+        {
+            "type": "waitAsync"
+        },
+        {
+            "type": "jump",
+            "from": [
+                7,
+                6
+            ],
+            "to": [
+                7,
+                6
+            ],
+            "time": 500,
+            "keep": true,
+            "async": true
+        },
+        {
+            "type": "move",
+            "loc": [
+                6,
+                3
+            ],
+            "time": 500,
+            "async": true,
+            "steps": [
+                "up:2"
+            ]
+        },
+        {
+            "type": "move",
+            "loc": [
+                7,
+                9
+            ],
+            "time": 500,
+            "async": true,
+            "steps": [
+                "up:1",
+                "right:3"
+            ]
+        },
+        {
+            "type": "setCurtain",
+            "color": [
+                0,
+                0,
+                0,
+                1
+            ],
+            "time": 1000,
+            "keep": true
+        },
+        {
+            "type": "waitAsync"
+        },
+        {
+            "type": "setBlock",
+            "number": "0",
+            "loc": [
+                [
+                    5,
+                    9
+                ],
+                [
+                    7,
+                    6
+                ],
+                [
+                    4,
+                    4
+                ],
+                [
+                    6,
+                    3
+                ]
+            ]
+        },
+        {
+            "type": "changeFloor",
+            "floorId": "guangchang2",
+            "loc": [
+                0,
+                0
+            ]
+        }
+    ],
     "eachArrive": [],
     "parallelDo": "",
     "events": {},
     "changeFloor": {
-        "12,6": {
-            "floorId": "street02",
-            "loc": [
-                0,
-                6
-            ],
-            "direction": "right"
-        },
         "6,12": {
-            "floorId": "guangchang2",
+            "floorId": "changguan1",
             "loc": [
                 6,
                 0
+            ]
+        },
+        "6,0": {
+            "floorId": "guangchang5",
+            "loc": [
+                6,
+                12
+            ]
+        },
+        "12,8": {
+            "floorId": "guangchang2",
+            "loc": [
+                0,
+                8
             ],
-            "direction": "down"
+            "direction": "right"
         }
     },
     "beforeBattle": {},
@@ -41,29 +290,65 @@ main.floors.guangchang=
     "cannotMove": {},
     "cannotMoveIn": {},
     "map": [
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0, 94],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0, 93,  0,  0,  0,  0,140,141]
+    [120007,120007,120007,120007,120020,  0, 91,  0,  0,130187,130188,130189,130190],
+    [120015,120015,120015,120015,120028,  0,  0,  0,  0,120144,120145,120146,120147],
+    [130172,130173,130174,130175,  0,  0,  0,  0,71320,120152,120153,120154,120155],
+    [130180,130181,130182,130183,200199,  0,  0,122,71328,120160,120161,120162,120163],
+    [130188,130189,130190,130191,121,  0,  0,  0,71336,120168,120169,120170,120171],
+    [130196,130197,130198,130199,140087,  0,121,  0,  0,120176,140020,140020,140020],
+    [130204,  0,130206,130207,140095,  0,  0,  0,  0,  0,140028,140028,140028],
+    [  0,  0,130214,130215,140103,  0,121,  0,  0,  0,  0,  0,  0],
+    [  0,  0,130214,120172,140111,  0,  0,  0,123,  0,  0,  0, 94],
+    [  0,  0,  0,120180,121,  0,  0,121,  0,200199,  0,  0,  0],
+    [140107,140108,140109,362,363,  0,  0,  0,361,362,362,362,362],
+    [210186,210186,210186,210186,210179,  0,  0,  0,210179,210185,210186,210186,210186],
+    [210178,210178,210178,210178,210171,  0, 93,  0,210171,210177,210178,210178,210178]
 ],
     "areas": "牢狱",
     "bgmap": [
-
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30055,30052,30052,30052,30052,30052,30055,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30055,30052],
+    [30052,30052,30052,30052,30055,30052,30052,30063,30062,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30055,30052,30062,30052,30052,30052,30052,30052,30052,30052,30055,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30062,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30062,30052],
+    [30052,30052,30052,30055,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [120177,120178,120179,30052,30052,30055,30052,30052,30055,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30062,30052,30052,30052,30052,30055,30052],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "fgmap": [
-
+    [  0,  0,  0,  0,200175,  0,  0,  0,  0,130187,130188,130189,130190],
+    [  0,  0,  0,  0,200183,  0,  0,  0,140086,130195,130196,130197,130198],
+    [  0,  0,  0,  0,200191,  0,  0,  0,140094,130203,130204,130205,130206],
+    [  0,  0,  0,  0,  0,  0,  0,  0,140102,130211,130212,130213,130214],
+    [  0,  0,  0,  0,  0,  0,  0,  0,140110,  0,140091,140092,140093],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140099,140100,140101],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,200175,140107,140108,140109],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,200183,  0,  0,  0],
+    [140091,140092,140093,  0,  0,  0,  0,  0,  0,200191,  0,  0,  0],
+    [140099,140100,140101,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [362,362,362,  0,  0,  0,  0,  0,  0,  0,  0,110242,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,110249,110250,110251],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,110256,110257,110258,110259]
 ],
     "bg2map": [
-
+    [200195,200196,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [90652,80106,90653,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [120145,120146,120147,  0,  0,  0,  0,  0,  0,  0,120177,120178,120179],
+    [120153,120154,120155,120156,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [120161,120162,120163,120164,90642,  0,  0,  0,  0,  0,  0,  0,  0],
+    [120169,120170,120171,  0,90642,  0,  0,  0,  0,  0,  0,  0,  0],
+    [130007,130007,130007,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [130015,130015,130015,  0,  0,  0,  0,  0,110239,110239,110239,110239,110239],
+    [  0,  0,  0,  0,130046,130046,130046,130046,130046,  0,  0,  0,  0],
+    [  0,  0,  0,  0,130046,130046,130046,130046,130046,  0,  0,  0,  0]
 ],
     "fg2map": [
 
diff --git a/project/floors/guangchang2.js b/project/floors/guangchang2.js
index 8247ca4..2d9947a 100644
--- a/project/floors/guangchang2.js
+++ b/project/floors/guangchang2.js
@@ -10,25 +10,52 @@ main.floors.guangchang2=
     "canUseQuickShop": true,
     "images": [],
     "ratio": 1,
-    "defaultGround": "grass2",
-    "firstArrive": [],
+    "defaultGround": "X90078",
+    "firstArrive": [
+        {
+            "type": "setCurtain",
+            "time": 1000
+        },
+        {
+            "type": "sleep",
+            "time": 2000
+        },
+        {
+            "type": "setCurtain",
+            "color": [
+                0,
+                0,
+                0,
+                1
+            ],
+            "time": 1000,
+            "keep": true
+        },
+        {
+            "type": "changeFloor",
+            "floorId": "guangchang5",
+            "loc": [
+                0,
+                0
+            ]
+        }
+    ],
     "eachArrive": [],
     "parallelDo": "",
     "events": {},
     "changeFloor": {
-        "6,0": {
+        "0,8": {
             "floorId": "guangchang",
             "loc": [
-                6,
-                12
-            ],
-            "direction": "up"
+                12,
+                8
+            ]
         },
-        "12,10": {
-            "floorId": "xiaoxiang01",
+        "2,0": {
+            "floorId": "guangchang6",
             "loc": [
-                0,
-                10
+                2,
+                12
             ]
         }
     },
@@ -40,29 +67,65 @@ main.floors.guangchang2=
     "cannotMove": {},
     "cannotMoveIn": {},
     "map": [
-    [  0,  0,  0,  0,  0,  0, 91,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,141],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0, 94],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140]
+    [  0,  0, 91,  0,210184,210186,210186,210186,210186,210186,210186,210186,143],
+    [  0,  0,  0,  0,210192,210194,210194,210175,210194,210194,210194,210194,143],
+    [120156,  0,  0,  0,90664,90648,90648,  0,90644,90648,90649,  0,143],
+    [120164,  0,  0,  0,90665,140110,122,  0,90645,140110,122,  0,120120],
+    [120172,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,120136],
+    [120180,  0,  0,  0,90656,90648,90649,  0,90636,90649,90648,  0,120137],
+    [  0,  0,  0,  0,90657,140110,122,  0,90638,140110,122,  0,120145],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,120160],
+    [ 92,  0,  0,  0,110384,110385,110376,  0,90652,90648,90649,  0,120168],
+    [  0,  0,  0,  0,110384,  0,122,  0,90653,  0,122,  0,120176],
+    [362,362,363,  0,  0,  0,  0,  0,  0,  0,  0,  0,90266],
+    [210181,210181,210181,  0,  0,  0,  0,  0,  0,  0,  0,  0,110189],
+    [210186,210186,210186,361,362,362,362,362,362,362,362,362,110197]
 ],
     "areas": "牢狱",
     "bgmap": [
-
+    [30052,30052,30052,30052,30052,30062,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30055,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30062,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30062,30052,30052,30052,30052,30052,30052],
+    [30052,30055,30052,30052,30052,30055,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30062,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30055,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [210209,210209,210209,30052,30052,30052,30062,30052,30055,30052,30052,30052,30052],
+    [210209,210209,210209,30062,30052,30052,30052,30052,30052,30052,30052,30052,30052]
 ],
     "fgmap": [
-
+    [130191,  0,  0,  0,  0,140086,140087,  0,  0,140086,140087,  0,  0],
+    [130199,140087,  0,  0,  0,140094,140095,  0,  0,140094,140095,  0,  0],
+    [130207,140095,  0,  0,  0,140102,140103,  0,  0,140102,140103,  0,  0],
+    [130215,140103,  0,  0,  0,140086,140087,  0,  0,140086,140087,  0,  0],
+    [  0,140111,  0,  0,  0,140094,140095,  0,  0,140094,140095,  0,  0],
+    [  0,  0,  0,  0,  0,140102,140103,  0,  0,140102,140103,  0,  0],
+    [  0,  0,  0,  0,  0,140086,140087,  0,  0,140086,140087,  0,  0],
+    [  0,  0,  0,  0,  0,140094,140095,  0,  0,140094,140095,  0,  0],
+    [  0,  0,  0,  0,  0,140102,140103,  0,  0,140102,140103,  0,  0],
+    [  0,  0,  0,  0,  0,140110,140111,  0,  0,140110,140111,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "bg2map": [
-
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,130192],
+    [  0,  0,  0,  0,  0,140110,140111,  0,  0,140110,140111,  0,130200],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,140110,140111,  0,  0,140110,140111,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [110239,110239,110239,364,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,364,  0,  0,  0,  0,  0,  0,  0,  0,90266],
+    [  0,  0,  0,110239,110239,110239,110239,110239,110239,110239,110239,110239,  0]
 ],
     "fg2map": [
 
diff --git a/project/floors/guangchang2_1.js b/project/floors/guangchang2_1.js
new file mode 100644
index 0000000..c460473
--- /dev/null
+++ b/project/floors/guangchang2_1.js
@@ -0,0 +1,98 @@
+main.floors.guangchang2_1=
+{
+    "floorId": "guangchang2_1",
+    "title": "广场",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "6,12": {
+            "floorId": "guangchang",
+            "loc": [
+                6,
+                0
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [210094,210095,210144,210145,210145,210145,210145,210145,210145,210145,210146,210092,210093],
+    [210102,210103,210152,210153,210153,210153,210153,210153,210153,210153,210154,210100,210101],
+    [210110,210111,210160,210161,210161,210161,210161,210161,210161,210161,210162,210108,210109],
+    [210118,210119,210168,210169,210169,210169,210169,210169,210169,210169,210170,210116,210117],
+    [210126,210127,210181,210181,210176,210183,210181,210181,210176,210183,210181,210124,210125],
+    [362,363,  0,210178,210184,210177,210178,210178,210184,210177,  0,361,362],
+    [210181,210181,210181,210178,210184,210177,210178,210178,210184,210177,210181,210181,210181],
+    [210178,210178,210178,210178,210184,210185,210186,210186,210192,210177,210178,210178,210178],
+    [210194,210194,210194,210194,210192,210193,210150,210194,210192,210193,210194,210194,210194],
+    [201037,201037,201037,201029,  0,  0,  0,  0,  0,201029,201037,201037,201037],
+    [210070,210071,210002,201029,  0,  0,  0,  0,  0,201029,210002,210068,210069],
+    [210078,210079,210002,201029,  0,  0, 93,  0,  0,201029,210002,210076,210077]
+],
+    "areas": "牢狱",
+    "bgmap": [
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [180025,180025,  0,  0,  0,  0,  0,  0,  0,  0,  0,180025,180025],
+    [180025,180025,  0,  0,  0,  0,  0,  0,  0,  0,  0,180025,180025],
+    [180025,180025,  0,  0,  0,  0,  0,  0,  0,  0,  0,180025,180025],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [190000,190000,190000,190000,210209,210209,210209,210209,210209,190000,190000,190000,190000],
+    [190000,190000,190000,190000,210209,210209,210209,210209,210209,190000,190000,190000,190000],
+    [190000,190000,190000,190000,210209,210209,210209,210209,210209,190000,190000,190000,190000],
+    [190000,190000,190000,190000,210209,210209,210209,210209,210209,190000,190000,190000,190000]
+],
+    "fgmap": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,210134,  0,  0,  0,210134,  0,  0,  0,  0],
+    [  0,  0,  0,  0,210142,  0,  0,  0,210142,  0,  0,  0,  0],
+    [  0,  0,210142,  0,210150,  0,210130,  0,210150,  0,210142,  0,  0],
+    [  0,  0,210150,  0,  0,  0,210138,  0,  0,  0,210150,  0,  0],
+    [  0,  0,  0,  0,  0,  0,210142,  0,  0,  0,  0,  0,  0],
+    [201021,201021,201021,201021,  0,  0,  0,  0,  0,201021,201021,201021,201021],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,210206,  0,210194,  0,210207,  0,  0,  0,  0],
+    [  0,  0,  0,  0,210208,  0,  0,  0,210210,  0,  0,  0,  0],
+    [210002,210002,  0,  0,210208,  0,  0,  0,210210,  0,  0,210002,210002],
+    [  0,  0,  0,  0,210208,  0,  0,  0,210210,  0,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/guangchang2_2.js b/project/floors/guangchang2_2.js
new file mode 100644
index 0000000..a70e8c9
--- /dev/null
+++ b/project/floors/guangchang2_2.js
@@ -0,0 +1,105 @@
+main.floors.guangchang2_2=
+{
+    "floorId": "guangchang2_2",
+    "title": "广场",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "7,0": {
+            "floorId": "guangchang3",
+            "loc": [
+                7,
+                12
+            ]
+        },
+        "6,12": {
+            "floorId": "guangchang2",
+            "loc": [
+                6,
+                0
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  0,  0,  0,  0,  0,  0,  0, 91,  0,  0,  0,  0,120160],
+    [210094,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,120168],
+    [210102,210103,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,120176],
+    [210110,210111,110231,110231,110231,  0,  0,  0,110231,110231,110231,110231,110231],
+    [210118,365,362,362,363,  0,  0,  0,361,362,362,362,362],
+    [210126,365,210177,210178,210176,  0,  0,  0,210176,210177,210178,210178,210178],
+    [362,363,210070,210071,210184,  0,  0,  0,210184,210068,210069,210178,210178],
+    [210181,210184,210078,210079,210192,  0,  0,  0,210192,210076,210077,210194,210194],
+    [210178,210184,210086,210087,  0,  0,  0,  0,  0,210084,210085,110239,110239],
+    [210194,210192,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [201037,201037,201029,  0,  0,  0,  0,  0,  0,130171,130172,130173,130174],
+    [210002,210002,201029,  0,  0,  0,  0,  0,  0,130179,130180,130181,130182],
+    [210002,210002,201029,  0,  0,  0, 93,  0,  0,130187,130188,130189,130190]
+],
+    "areas": "牢狱",
+    "bgmap": [
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [180025,180025,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [180025,180025,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [180025,180025,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [190000,190000,190000,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [190000,190000,190000,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [190000,190000,190000,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209],
+    [190000,190000,190000,210209,210209,210209,210209,210209,210209,210209,210209,210209,210209]
+],
+    "fgmap": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,110231,110231],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [201021,201021,201021,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,120160],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,120168],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,120176],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,210119,110239,110239,110239,  0,  0,  0,110239,110239,110239,110239,110239],
+    [  0,210127,  0,  0,  0,130046,130046,130046,  0,  0,  0,  0,  0],
+    [  0,  0,210177,210178,  0,130046,130046,130046,  0,210177,210178,  0,  0],
+    [  0,  0,210193,210194,  0,130054,130054,130054,  0,210193,210194,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [210217,210217,210217,210206,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,210208,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,210208,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,210208,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/guangchang3.js b/project/floors/guangchang3.js
new file mode 100644
index 0000000..06fa931
--- /dev/null
+++ b/project/floors/guangchang3.js
@@ -0,0 +1,132 @@
+main.floors.guangchang3=
+{
+    "floorId": "guangchang3",
+    "title": "主塔 10 层",
+    "name": "10",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [
+        {
+            "type": "setCurtain",
+            "time": 1000
+        },
+        {
+            "type": "sleep",
+            "time": 2000
+        },
+        {
+            "type": "setCurtain",
+            "color": [
+                0,
+                0,
+                0,
+                1
+            ],
+            "time": 1000,
+            "keep": true
+        },
+        {
+            "type": "changeFloor",
+            "floorId": "guangchang4",
+            "loc": [
+                0,
+                0
+            ]
+        }
+    ],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "0,9": {
+            "floorId": "guangchang4",
+            "loc": [
+                12,
+                9
+            ]
+        },
+        "6,12": {
+            "floorId": "guangchang5",
+            "loc": [
+                6,
+                0
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [210161,210161,210162,210185,210186,210186,210186,210186,210186,210186,210160,210161,210161],
+    [210169,210169,210170,210185,210186,210186,210186,210186,210186,210186,210168,210169,210169],
+    [210178,210178,210176,210193,210194,210194,210194,210194,210194,210194,210176,210177,210178],
+    [210178,210167,210184,  0,  0,  0,  0,  0,  0,  0,210184,210159,210178],
+    [210178,210178,210184,  0,  0,  0,132,  0,  0,  0,210184,210177,210178],
+    [210186,210186,210184,362,363,  0,  0,  0,361,362,210184,210185,210186],
+    [210194,210194,210192,210186,210186,  0,  0,  0,210186,210186,210184,210193,210194],
+    [71795,71814,210192,71655,210194,  0,  0,  0,210194,210194,210192,50165,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,71421,10168,71784,71785],
+    [ 92,  0,  0,  0,  0,  0,  0,  0,  0,  0,10176,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,122,  0,  0],
+    [210145,210145,210146,  0,  0,  0,  0,  0,  0,  0,  0,71799,  0],
+    [210153,210153,210154,  0,  0,  0, 93,  0,  0,  0,90648,90657,90850]
+],
+    "bgmap": [
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,210209,210209,210209,210209,210209,210209,210209,30052,30052,30052],
+    [30052,30052,30052,210209,210209,210209,210209,210209,210209,210209,30052,30052,30052],
+    [30052,30052,30052,210209,210209,210209,210209,210209,210209,210209,30052,30052,30052],
+    [30052,30052,30052,210209,210209,210209,210209,210209,210209,210209,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30062,30052,30052,30052,30052,30052,30052,30052,30052,30052,30055,30052],
+    [30052,30052,30052,30052,30055,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30062,30052,30052,30052,30062,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30055,30052,30052,30052],
+    [30052,30055,30052,30052,30052,30062,30052,30052,30052,30052,30052,30052,30052]
+],
+    "fgmap": [
+    [  0,  0,  0,  0,210142,  0,  0,  0,210142,  0,  0,  0,  0],
+    [  0,  0,  0,  0,210150,  0,  0,  0,210150,  0,  0,  0,  0],
+    [  0,210142,  0,  0,  0,  0,  0,  0,  0,  0,  0,210142,  0],
+    [210131,210150,  0,  0,  0,  0,  0,  0,  0,  0,  0,210150,210131],
+    [210139,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,210139],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [71787,71806,  0,  0,  0,  0,  0,  0,  0,  0,  0,50157,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,71413,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140086],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,50151,71217,140094],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,71791,140102],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140110],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140086]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,130046,130046,130046,  0,  0,  0,  0,  0],
+    [  0,  0,  0,210194,  0,130054,130054,130054,  0,210194,  0,  0,  0],
+    [  0,  0,  0,71663,  0,  0,  0,  0,90849,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,71792,71793],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,71225,71328],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,71336],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/guangchang4.js b/project/floors/guangchang4.js
new file mode 100644
index 0000000..cb6d123
--- /dev/null
+++ b/project/floors/guangchang4.js
@@ -0,0 +1,138 @@
+main.floors.guangchang4=
+{
+    "floorId": "guangchang4",
+    "title": "主塔 10 层",
+    "name": "10",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [
+        {
+            "type": "setCurtain",
+            "time": 1000
+        },
+        {
+            "type": "autoText",
+            "text": "是被险峻的峭壁环绕,与世隔绝的,都市的最底部",
+            "time": 2000
+        },
+        {
+            "type": "setCurtain",
+            "color": [
+                0,
+                0,
+                0,
+                1
+            ],
+            "time": 1000,
+            "keep": true
+        },
+        {
+            "type": "changeFloor",
+            "floorId": "changguan1",
+            "loc": [
+                0,
+                0
+            ]
+        }
+    ],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "12,9": {
+            "floorId": "guangchang3",
+            "loc": [
+                0,
+                9
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [210161,210161,210161,210162,364,271,  0,271,365,210160,210161,210161,210161],
+    [210169,210169,210169,210170,364,  0,272,  0,365,210168,210169,210169,210169],
+    [210181,210181,210182,210176,361,362,362,362,363,210176,210180,210181,210181],
+    [210178,210178,210159,210184,210185,210186,210186,210186,210186,210184,210177,210178,210178],
+    [210178,210178,210178,210184,210177,10104,10105,10106,210167,210184,210177,210178,210178],
+    [210186,210186,210186,210184,210193,10112,10113,10114,210194,210184,210185,210186,210186],
+    [71834,71835,210194,210192,30167,  0,  0,  0,30167,210192,210175,210194,210194],
+    [130173,71835,130175,  0,  0,  0,  0,  0,  0,  0,71799,71799,  0],
+    [130181,130182,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [130189,130190,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0, 94],
+    [130189,130190,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [130189,130190,  0,  0,  0,  0,  0,  0,  0,  0,210144,210145,210145],
+    [130197,130198,  0,90601,90601,90601,90601,90601,90601,90601,210152,210153,210153]
+],
+    "bgmap": [
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30062,30052,30052,30052,30052,30052,30055,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30062,30052,30052],
+    [30052,30052,30052,30062,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30055,30052,30052,30052,30052,30052,30052,30062,30052,30052,30052,30055,30052],
+    [30052,30052,30052,30052,30055,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30055,30052,30052,30052,30052,30052,30052,30052,30052,30052,30062]
+],
+    "fgmap": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,210142,  0,  0,  0,  0,  0,  0,  0,210142,  0,  0],
+    [210131,210131,210150,  0,210142,  0,  0,  0,210142,  0,210150,210131,210131],
+    [210139,210139,  0,  0,210150,  0,  0,  0,210150,  0,  0,210139,210139],
+    [210142,  0,  0,  0,30159,  0,  0,  0,30159,  0,  0,  0,210142],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,71791,71791,71786],
+    [  0,130174,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,71794],
+    [  0,  0,130183,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,130191,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,130191,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,130191,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,130191,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,210212,210212,210212,210212,210212,  0,  0,  0,  0],
+    [  0,  0,  0,  0,210212,210212,210212,210212,210212,  0,  0,  0,  0],
+    [  0,  0,  0,  0,210212,210212,210212,210212,210212,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,210178,210178,210178,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,210194,210194,210194,  0,  0,  0,  0,  0],
+    [210150,210194,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [71842,71843,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [120146,120147,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,210150],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+]
+}
\ No newline at end of file
diff --git a/project/floors/guangchang5.js b/project/floors/guangchang5.js
new file mode 100644
index 0000000..86715ef
--- /dev/null
+++ b/project/floors/guangchang5.js
@@ -0,0 +1,141 @@
+main.floors.guangchang5=
+{
+    "floorId": "guangchang5",
+    "title": "广场",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [
+        {
+            "type": "setCurtain",
+            "time": 1000
+        },
+        {
+            "type": "autoText",
+            "text": "《特别受灾地区》——",
+            "time": 2000
+        },
+        {
+            "type": "setCurtain",
+            "color": [
+                0,
+                0,
+                0,
+                1
+            ],
+            "time": 1000,
+            "keep": true
+        },
+        {
+            "type": "changeFloor",
+            "floorId": "guangchang3",
+            "loc": [
+                0,
+                0
+            ]
+        }
+    ],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "6,12": {
+            "floorId": "guangchang",
+            "loc": [
+                6,
+                0
+            ]
+        },
+        "6,0": {
+            "floorId": "guangchang3",
+            "loc": [
+                6,
+                12
+            ]
+        },
+        "12,9": {
+            "floorId": "guangchang6",
+            "loc": [
+                0,
+                9
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [210161,210157,210162,  0,  0,  0, 91,  0,  0,  0,  0,  0,140094],
+    [210164,210165,210170,  0,  0,  0,  0,  0,  0,  0,  0,20110,140094],
+    [210162,210178,210179,  0,  0,  0,  0,  0,  0,  0,122,  0,140102],
+    [210170,210178,210171,  0,  0,  0,  0,  0,  0,  0,  0,20110,140110],
+    [  0,365,362,362,363,  0,  0,  0,361,362,362,362,362],
+    [  0,365,210177,210178,210176,  0,  0,  0,210176,210167,210178,210178,210178],
+    [362,363,210070,210071,210184,  0,  0,  0,210184,210068,210069,210178,210166],
+    [210181,210184,210078,210079,210192,  0,  0,  0,210192,210076,210077,210194,210194],
+    [210178,210184,210086,210087,  0,  0,  0,  0,  0,210084,210085,110239,110239],
+    [210194,210192,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0, 94],
+    [200171,200172,120031,120031,120020,  0,  0,  0,  0,  0,  0,  0,  0],
+    [200179,200180,122,  0,  0,  0,  0,  0,200199,  0,130180,130181,130182],
+    [200187,200188,  0,  0,  0,  0, 93,  0,  0,130187,130188,130189,130190]
+],
+    "areas": "牢狱",
+    "bgmap": [
+    [30052,30052,30055,30052,30052,30052,30052,30052,30052,30062,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30055,30052],
+    [30055,30052,30052,30052,30052,30052,30055,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30062,30052,30052,30052,30052,30052,30052,30062,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30055,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [210209,210209,30052,30052,30052,30052,30062,30052,30052,30052,30052,30052,30052],
+    [210209,210209,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30052],
+    [30052,30052,30052,30052,30055,30052,30052,30052,30052,30052,30052,30052,30055],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30052,30052,30062,30052,30052],
+    [30052,30052,30052,30052,30052,30052,30052,30052,30055,30052,30052,30052,30052],
+    [30052,30052,30052,30062,30052,30052,30052,30052,30052,30052,30052,30052,30052]
+],
+    "fgmap": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,20102,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,20102,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,120080,  0,  0,  0,  0,  0,  0,  0,  0,120080,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [200115,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,110231,110231],
+    [200147,200124,  0,  0,  0,  0,  0,  0,200175,  0,  0,  0,  0],
+    [200147,200148,120026,120026,120004,  0,  0,  0,200183,  0,  0,  0,  0],
+    [200155,200156,  0,  0,  0,  0,  0,  0,200191,  0,130172,130173,130174],
+    [200163,200164,  0,  0,  0,  0,  0,  0,  0,130179,  0,  0,  0],
+    [  0,  0,  0,  0,120004,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,210178,210179,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,210178,210195,  0,  0,  0,  0,  0,  0,  0,  0,  0,140110],
+    [210179,210178,110239,110239,110239,  0,  0,  0,110239,110239,110239,110239,110239],
+    [210171,  0,  0,  0,  0,130046,130046,130046,  0,  0,  0,  0,  0],
+    [210195,  0,210177,210178,  0,130046,130046,130046,  0,210177,210178,  0,  0],
+    [  0,  0,210175,210194,  0,130054,130054,130054,  0,210193,210194,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,200173,  0,120012,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,200181,  0,120028,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,200189,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/guangchang6.js b/project/floors/guangchang6.js
new file mode 100644
index 0000000..e2f79e3
--- /dev/null
+++ b/project/floors/guangchang6.js
@@ -0,0 +1,116 @@
+main.floors.guangchang6=
+{
+    "floorId": "guangchang6",
+    "title": "广场",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "2,12": {
+            "floorId": "guangchang2",
+            "loc": [
+                2,
+                0
+            ]
+        },
+        "0,9": {
+            "floorId": "guangchang5",
+            "loc": [
+                12,
+                9
+            ]
+        },
+        "12,7": {
+            "floorId": "yiqu1",
+            "loc": [
+                0,
+                7
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {
+        "4,12": [],
+        "4,11": [],
+        "4,10": []
+    },
+    "map": [
+    [140095,120160,120161,120162,120163,120164,365,  0,  0,  0,  0,143,143],
+    [140095,120168,120169,120170,120171,120172,365,  0,  0,  0,  0,143,143],
+    [140103,120176,120177,20122,120179,120180,365,20122,122,  0,  0,143,143],
+    [140111,20110,140107,140108,140109,20110,365,20130,  0,  0,130075,130192,130193],
+    [362,362,362,362,362,362,362,20113,  0,  0,130082,130200,130201],
+    [210178,210178,210178,210167,210178,210178,210176,  0,  0,  0,137,140099,140100],
+    [210178,210178,210178,210178,210178,210178,210184,20113,121,  0,  0,140107,140108],
+    [210194,210175,210194,210194,210194,210194,210192,20114,  0,  0,  0,  0, 94],
+    [110239,  0,  0,  0,  0,  0,365,  0,  0,  0,  0,120109,120110],
+    [ 92,  0,  0,  0,362,  0,363,128,  0,  0,10165,120117,120118],
+    [  0,  0,  0,  0, 17,  0, 17,  0,  0,  0,10173,120125,120126],
+    [130183,  0,  0,  0, 17,  0, 17,  0,  0,  0,10173,120133,120134],
+    [130191,  0, 93,  0, 17,  0,  0,  0,  0,138,10157,120141,120142]
+],
+    "areas": "牢狱",
+    "bgmap": [
+    [30055,30052,30052,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30055,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,90078,  0,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,  0,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,30052,30052,30052,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,210187,210178,210187,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,210195,210178,210195,90078,90078,90078,90078,90078,90078],
+    [30052,30052,30052,30052,90078,90078,90078,90078,90078,90078,90078,90078,90078]
+],
+    "fgmap": [
+    [  0,  0,  0,  0,  0,  0,  0,140087,  0,  0,140094,  0,  0],
+    [  0,  0,140091,140092,140093,  0,  0,140095,  0,  0,140102,  0,  0],
+    [  0,20102,140099,140100,140101,20102,  0,140103,  0,  0,140110,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,140111,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,140087,  0,  0,  0,140091,140092],
+    [  0,  0,  0,  0,  0,  0,  0,140095,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,140103,  0,  0,  0,  0,  0],
+    [110231,  0,  0,  0,  0,  0,365,140111,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140086,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140094,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140102,  0,  0]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,160032,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,160032,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,120178,  0,  0,160032,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,20130,  0,  0,160032,  0,  0,  0,130088,  0,  0],
+    [120103,120103,120103,120103,120103,120103,120103,  0,  0,  0,130096,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,130076,130076],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,130084,130084],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,365,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,365,130045,130046,130047,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,365,130045,130046,130047,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,365,130053,130054,130055,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/guangchangquan.js b/project/floors/guangchangquan.js
new file mode 100644
index 0000000..e5040e3
--- /dev/null
+++ b/project/floors/guangchangquan.js
@@ -0,0 +1,194 @@
+main.floors.guangchangquan=
+{
+    "floorId": "guangchangquan",
+    "title": "广场",
+    "name": "1",
+    "width": 26,
+    "height": 39,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {},
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [210178,210178,210178,210178,210171,  0,  0,  0,210171,210178,210178,210178,210178,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [210178,210178,210178,210178,210171,  0,  0,  0,210171,210178,210178,210178,210178,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "areas": "牢狱",
+    "bgmap": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,140061,140061,140061,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,140061,140061,140061,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0, 86, 86, 86, 86,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0, 86, 84, 84, 84, 84,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0, 86, 84, 84, 84, 84,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0, 86, 84, 84, 84, 84,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0, 86, 84, 84, 84, 84,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0, 86, 86, 86, 86,140061,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,140061,140061,140061,140061,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,140061,140061,140061,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,140061,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,20067,20067,20067,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+]
+}
\ No newline at end of file
diff --git a/project/floors/jiaocheng01.js b/project/floors/jiaocheng01.js
new file mode 100644
index 0000000..98b85b0
--- /dev/null
+++ b/project/floors/jiaocheng01.js
@@ -0,0 +1,285 @@
+main.floors.jiaocheng01=
+{
+    "floorId": "jiaocheng01",
+    "title": "教程1",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [
+        {
+            "type": "setHeroOpacity",
+            "opacity": 1
+        },
+        {
+            "type": "setCurtain",
+            "time": 100
+        }
+    ],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {
+        "0,11": [
+            "宝石介绍:攻防法速\n加点量:2/2/5/2"
+        ],
+        "12,11": [
+            "血瓶介绍:不同大小"
+        ],
+        "5,7": [
+            {
+                "type": "choices",
+                "text": "\t[小妖精,fairy]是否要进行钥匙和宝物方面的教学呢?",
+                "choices": [
+                    {
+                        "text": "我太需要了",
+                        "action": [
+                            "\t[小妖精,fairy]一种颜色的钥匙开一种颜色的门。\n一把钥匙只能开一扇门。",
+                            "\t[小妖精,fairy]除了钥匙以外,地图上常见的宝物还有宝石和血瓶。\n血瓶可以增加角色的生命值。\n血瓶的颜色、样式和所处的区域决定了血瓶增加的生命值数量。\n在第一个区域,四种颜色的基础血瓶分别可以回复角色100,\n250,500和900点生命值。",
+                            "\t[小妖精,fairy]宝石可以增加角色的四维。\n红宝石增加攻击力。\n蓝宝石增加防御力。\n绿宝石增加法术强度。\n黄宝石增加速度。",
+                            "\t[小妖精,fairy]不同样式、不同区域的宝石决定了宝石增加的能力幅度。\n在第一个区域,红、蓝、黄色的基础宝石可以增加角色对应能力2点。\n绿色的基础宝石可以增加角色法术强度5点。",
+                            {
+                                "type": "jump",
+                                "to": [
+                                    2,
+                                    2
+                                ],
+                                "time": 500,
+                                "keep": true
+                            }
+                        ]
+                    },
+                    {
+                        "text": "我已经了解了",
+                        "action": [
+                            {
+                                "type": "setValue",
+                                "name": "flag:跳过1",
+                                "value": "1"
+                            },
+                            {
+                                "type": "jump",
+                                "to": [
+                                    6,
+                                    1
+                                ],
+                                "time": 500,
+                                "keep": true
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        2,
+                                        2
+                                    ]
+                                ],
+                                "remove": true
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        8,
+                                        1
+                                    ]
+                                ],
+                                "remove": true
+                            }
+                        ]
+                    }
+                ]
+            }
+        ],
+        "2,2": [
+            "\t[小妖精,fairy]在这个游戏中,不同颜色的钥匙珍贵程度也不相同。",
+            "\t[小妖精,fairy]黄色的钥匙最为常见。",
+            "\t[小妖精,fairy]蓝色的钥匙有点罕见,其价值在正式游戏中略高于2把黄钥匙,但低于3把黄钥匙。",
+            "\t[小妖精,fairy]红色的钥匙比较稀有,在正式游戏中一般一把红钥匙可以换取3把蓝钥匙,但反之则有价无市。\n红色的钥匙在正式游戏中一般仅用于boss战相关,开启非常重要的道具,或绕过非常非常强力的怪物。",
+            "\t[小妖精,fairy]绿色的钥匙为剧情道具,在游戏进程中绿钥匙和不可绕过的绿门一一对应,但存在通往相同区域的二选一绿门。\n绿色的钥匙一般被重兵把守,且不参与其他钥匙的相互兑换。",
+            {
+                "type": "jump",
+                "to": [
+                    8,
+                    1
+                ],
+                "time": 500,
+                "keep": true
+            }
+        ],
+        "8,1": [
+            "\t[小妖精,fairy]在地图上,偶尔会出现这种看起来白送一把蓝钥匙的房间。",
+            "\t[小妖精,fairy]然而,拿到这把“免费的”蓝钥匙,需要一把红钥匙作为前置。\n在教程关卡中,你肯定拥有这把红钥匙——如果你没有立即拿取旁边绿钥匙的话。",
+            "\t[小妖精,fairy]在正式游戏中,你可能当时没有,甚至在一时半会儿都不会有这把红钥匙。\n那样的话,这把蓝钥匙只能寄存在房间里。\n在拿到红钥匙之后,你的第一优先级行动就是回来拿上这把蓝钥匙。",
+            "\t[小妖精,fairy]这把红钥匙,以及用1红钥匙换取1红1蓝钥匙的操作,称为钥匙的周转。",
+            "\t[小妖精,fairy]当然,如果门口有怪物把守的话就另当别论——等到损失比较合适的时候再回来吧。",
+            {
+                "type": "jump",
+                "to": [
+                    6,
+                    1
+                ],
+                "time": 500,
+                "keep": true
+            },
+            {
+                "type": "setValue",
+                "name": "flag:教程1",
+                "value": "1"
+            }
+        ],
+        "6,1": [
+            {
+                "type": "if",
+                "condition": "((Math.max(flag:教程1,flag:跳过1))===1)",
+                "true": [
+                    "\t[小妖精,fairy]接下来就要开始适应本游戏的战斗系统了——",
+                    "\t[小妖精,fairy]在此之前,让我先确认一下吧。\n你拥有黄钥匙和蓝钥匙各一把,对吗?",
+                    "\t[小妖精,fairy]相对于其他RPG游戏来说,本作的战斗系统并不需要玩家在战斗中爆手速进行操作。\n暂停或挂机也是没问题的——怪物并不会因为现实中的你不操作而主动攻击你。\n换句话说,战斗的结果在战斗开始的一瞬间就已经确定了。",
+                    "\t[小妖精,fairy]除此之外,通过存档读档刷取随机数的操作在本作中也是不可行的。\n本作品将不涉及任何随机数的判定。\n这意味着如果你愿意,你可以百分百精确地复制一条攻略路线。",
+                    "\t[小妖精,fairy]换句话说,本作的战斗过程只依赖于战斗双方的各项数据。\n这包括双方的面板数值,各种技能,和多样的装备。",
+                    "\t[小妖精,fairy]本作还有另外一个特点——地图上的资源是有限的,并不会随时间自动刷新。\n这意味着不能在低级图反复刷取资源来碾压高级的怪物,必须仔细规划资源的获取及使用。",
+                    "\t[小妖精,fairy]在手部乘区、脸部乘区、肝部乘区都被限制住的情况下——\n请开发自己的脑部乘区,迎接后面的挑战吧!",
+                    {
+                        "type": "choices",
+                        "text": "\t[小妖精,fairy]是否需要进行简单的战斗系统讲解呢?",
+                        "choices": [
+                            {
+                                "text": "我太需要了",
+                                "action": []
+                            },
+                            {
+                                "text": "我已经很会了",
+                                "action": [
+                                    {
+                                        "type": "hide",
+                                        "loc": [
+                                            [
+                                                9,
+                                                10
+                                            ]
+                                        ],
+                                        "floorId": "jiaocheng02",
+                                        "remove": true
+                                    },
+                                    {
+                                        "type": "hide",
+                                        "loc": [
+                                            [
+                                                11,
+                                                8
+                                            ]
+                                        ],
+                                        "floorId": "jiaocheng02",
+                                        "remove": true
+                                    },
+                                    {
+                                        "type": "hide",
+                                        "loc": [
+                                            [
+                                                11,
+                                                4
+                                            ]
+                                        ],
+                                        "floorId": "jiaocheng02",
+                                        "remove": true
+                                    },
+                                    {
+                                        "type": "hide",
+                                        "loc": [
+                                            [
+                                                7,
+                                                4
+                                            ]
+                                        ],
+                                        "floorId": "jiaocheng02",
+                                        "remove": true
+                                    },
+                                    {
+                                        "type": "hide",
+                                        "loc": [
+                                            [
+                                                5,
+                                                2
+                                            ]
+                                        ],
+                                        "floorId": "jiaocheng02",
+                                        "remove": true
+                                    },
+                                    {
+                                        "type": "setBlock",
+                                        "number": "fairy",
+                                        "loc": [
+                                            [
+                                                4,
+                                                9
+                                            ]
+                                        ],
+                                        "floorId": "jiaocheng02"
+                                    },
+                                    {
+                                        "type": "setValue",
+                                        "name": "flag:跳过2",
+                                        "value": "1"
+                                    }
+                                ]
+                            }
+                        ]
+                    },
+                    {
+                        "type": "hide",
+                        "remove": true
+                    }
+                ]
+            }
+        ]
+    },
+    "changeFloor": {
+        "6,0": {
+            "floorId": "jiaocheng02",
+            "loc": [
+                6,
+                12
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  1,  1,  1,  1,  1,  1, 91,  1,  1,  1,  1,  1,  1],
+    [  1, 22,  0,  1, 22,  1,  0,  1,  0,  0, 23, 22,  1],
+    [  1, 21,  0,  1, 81,  1, 84,  1, 83,  1,  1,  1,  1],
+    [  1, 82,  1,  1,  0,  0,  0, 81,  0, 83,  0, 24,  1],
+    [  1,  0,  0, 82,  0,  1, 81,  1, 82,  1,  1,  1,  1],
+    [  1, 21,  1,  1,  0,  1, 22,  1,  0,  0, 21, 23,  1],
+    [  1,  1,  1,  1, 83,  1,  1,  1,  1,  1,  1,  1,  1],
+    [401,  0,402,  1,  0,124, 82,  0, 22,  1, 31,  0, 32],
+    [  0,  0,  0,  1, 21,  1,  1,  1,  0,  1,  0,  0,  0],
+    [403,  0,404,  1, 23,  1, 21,  1,  0,  1, 34,  0, 33],
+    [  1,  1,  1,  1,  1,  1,  0,  1, 81,  1,  1,  1,  1],
+    [129,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,129],
+    [  1,  1,  1,  1,  1,  1,  0,  1,  1,  1,  1,  1,  1]
+],
+    "areas": "教程",
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/jiaocheng02.js b/project/floors/jiaocheng02.js
new file mode 100644
index 0000000..7990160
--- /dev/null
+++ b/project/floors/jiaocheng02.js
@@ -0,0 +1,363 @@
+main.floors.jiaocheng02=
+{
+    "floorId": "jiaocheng02",
+    "title": "教程2",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {
+        "9,10": [
+            "\t[小妖精,fairy]恭喜你!开始了接触本作战斗系统的第一步!\n希望你刚才有好好看过战斗动画哦——当然没有也没关系!我们一点点教!",
+            "\t[小妖精,fairy]刚才我们的剑砍到敌人身上的时候,敌人身上冒出了一个“-16”的数字。\n这意味着我们砍出去的剑对敌人造成了16点伤害。\n这个“-16”是怎么来的呢?",
+            "\t[小妖精,fairy]打开怪物手册!",
+            {
+                "type": "callBook"
+            },
+            "\t[小妖精,fairy]我们拥有26点攻击力——在左边的状态栏可以看到。\n敌人拥有10点防御力——在怪物手册可以看到。\n我们的每次普通攻击将对怪物造成26-10=16点伤害。",
+            "\t[小妖精,fairy]同理,怪物的每次普通攻击将对我们造成46-16=30点伤害。\n在战斗动画中也有体现哦!",
+            "\t[小妖精,fairy]一方的生命值降低到0,意味着这一方被对方击杀。\n换句话说,如果我们的生命值被怪物的攻击压低到了0,这意味着勇士在战斗中牺牲了。\n(也就是game over啦)。",
+            "\t[小妖精,fairy]虽然在相同的状态下,持有更多的生命值总是更好,但生命值只是一个“够用就行”的属性。\n相较于把自己的生命值把握在自我设定的“500点”“1000点”等“安全线”上,\n用这些多余的生命值换取钥匙、能力,甚至是金币等资源,对后续的游戏过程来说会更轻松。",
+            "\t[小妖精,fairy]话说回来。\n怪物的生命值一共有222点。\n我们想要击杀怪物需要攻击怪物——222÷16(向上取整)=14次。\n为什么是向上取整?因为哪怕怪物还剩1点生命值,没死就是没死。",
+            "\t[小妖精,fairy]在我们攻击怪物14次的过程中,敌人也反击了我们13次。\n我们的生命因此损失了13*30=390点。\n如果没有关闭显伤开关的话,这个”390点“的数字将会显示在在怪物的左下角。\n剩余的生命仍然足够击败第二个相同的敌人呢——但好像还有什么事需要做?",
+            {
+                "type": "jump",
+                "to": [
+                    11,
+                    8
+                ],
+                "time": 500,
+                "keep": true
+            }
+        ],
+        "11,8": [
+            "\t[小妖精,fairy]在本作中,有一个便捷操作,叫做“自动存档”。\n具体点说,就是在我们与怪物发生战斗前,或开门前,都会进行一个存档。\n后面我们发现之前哪一步做错了,都有机会撤销,“反悔”——最多支持反悔50步之前的错误。\n读取自动存档的快捷键是“A”键。",
+            "\t[小妖精,fairy]红宝石可以提高角色的攻击力。\n不同形状、不同区域的红宝石,提高攻击力的程度也不一样。\n刚才黄门里的红宝石,在当前区域每一颗可以提升角色2点攻击力。",
+            {
+                "type": "if",
+                "condition": "(status:hp!==280)",
+                "true": [
+                    "\t[小妖精,fairy]一般来说,在地图上有无需战斗就可以获得的资源时,优先获得这些资源再进行战斗。\n获得资源提升能力,会让我们在战斗中更具有优势。\n不用为了后面保存黄钥匙!钥匙是够的!",
+                    "\t[小妖精,fairy]所以你为什么没有拿红宝石就急匆匆地来战斗了!",
+                    "\t[小妖精,fairy]请读取自动存档,回去拿完宝石再与敌人作战!"
+                ],
+                "false": [
+                    "\t[小妖精,fairy]在提升攻击力后,角色的攻击将更有力度。\n在获得2颗红宝石之后,角色每次攻击对怪物造成的伤害将由16点提升到20点。\n原本需要222÷16=14次攻击击杀的怪物,现在只需要222÷20=12次即可击杀。",
+                    "\t[小妖精,fairy]由于怪物受到更少次数的攻击就被击杀了,角色受到的攻击次数也减少了。\n从13次减少到了11次。",
+                    "\t[小妖精,fairy]但要注意——不是所有时候提升攻击力都能降低角色受到的攻击次数。\n如果角色的攻击力仅提升了1点,攻击伤害从16提升到17——\n击败怪物需要的攻击次数是222÷17,还是14次。这意味着战斗伤害不会有任何变化。",
+                    "\t[小妖精,fairy]因此,能带来战斗伤害减少的攻击力提升节点特别重要——我们称之为“临界”。\n从最初的26点攻击力,16点攻击伤害,需要14次攻击击杀,开始计算——\n222÷13=17.077,这意味着我们至少要达到17.077的攻击伤害,才能减少击杀敌人的攻击次数——但由于本作的攻击力是整数,你必须达到18的攻击伤害,也就是28点攻击力,才能降低战斗伤害。",
+                    "\t[小妖精,fairy]同理,以现在的状态开始计算。现在我们需要12次攻击击杀怪物。\n222÷11=20.182,我们需要21的攻击伤害,也就是31点攻击力。",
+                    {
+                        "type": "callBook"
+                    },
+                    "\t[小妖精,fairy]我们的攻击力距离下一个临界点还差1点;达到临界点可以让我们受到的伤害减少30点。\n在怪物手册中记作[1,30]或\"1:30\"。\n同样地,右上角相同的怪物身上,在战斗伤害330的数字上面,显示的是距离下一个攻击临界点的距离,\n1点。",
+                    "\t[小妖精,fairy]点击怪物手册中的特定怪物,可以查看怪物的详细信息。\n其中不仅包括怪物特殊属性的具体效果,还包含了详细的临界表格。\n在攻击临界表格内,“:”前面的数值表示还差多少攻击抵达接下来的临界,后面的数值表示抵达该临界可以在战斗中少受到多少伤害。",
+                    "\t[小妖精,fairy]但请注意,有的时候用更少的攻击次数击杀怪物,并不会让自己承受怪物的伤害变少哦!\n这就与我们接下来要讲解的属性,”速度“,有关啦。",
+                    "\t[小妖精,fairy]哎呀——生命值好像不太够了!\n捡起后面黄门里的血瓶,补充生命值继续后面的内容吧!",
+                    {
+                        "type": "jump",
+                        "to": [
+                            11,
+                            4
+                        ],
+                        "time": 500,
+                        "keep": true
+                    }
+                ]
+            }
+        ],
+        "11,4": [
+            "\t[小妖精,fairy]与其他魔塔游戏不同的是,本作不仅引入了“速度”属性,还引入了实时战斗系统。\n这将带来比回合制战斗更丰富,更具有策略性的战斗体验。",
+            "\t[小妖精,fairy]在本作中,“速度”意味着角色进行一次攻击前摇的快慢程度。没有特殊说明的话,本作不涉及攻击后摇。\n现在角色的速度为16,这意味着在单位时间内,角色可以攻击16次。\n战斗开始时,参战的双方互相向对方表露敌意,并开始分别计算攻击前摇。\n另一方面,速度和攻击前摇挂钩,意味着角色下一次攻击的时刻以上一次攻击落下时的速度为准;在此之后受到任何额外的减速效果,都只会影响到再下一次攻击。",
+            "\t[小妖精,fairy]在刚才的战斗中,怪物的速度为18,角色的速度为16。\n因此在互相展现敌意后,怪物的攻击先命中角色,随后才是角色对怪物造成伤害。\n此后双方一直匀速造成伤害。",
+            "\t[小妖精,fairy]战斗结束于8次攻击/16速度=0.5时刻。\n此时角色攻击8次,完成对怪物的击杀;与此同时,怪物的第9次攻击前摇刚好准备完毕。\n在本作的战斗系统中,如果怪物死亡时刻正好作出一次攻击(时刻完全相同)的话,视为这次攻击有效!\n因此,角色在之前的战斗中受到怪物刚好9次伤害。",
+            "\t[小妖精,fairy]接下来,请获取黄门中的2颗黄宝石,将角色的速度从16提升到20。\n面对相同的敌人,提升速度又会展现什么效果呢?",
+            {
+                "type": "jump",
+                "to": [
+                    7,
+                    4
+                ],
+                "time": 500,
+                "keep": true
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "7,4": [
+            {
+                "type": "if",
+                "condition": "(status:hp!==380)",
+                "true": [
+                    "\t[小妖精,fairy]都告诉你要获取黄宝石啦!地图上的生命资源并不是那么充足!再浪费的话后面肯定不够啦!",
+                    "\t[小妖精,fairy]请读取自动存档,回去拿完宝石再与敌人作战!"
+                ],
+                "false": [
+                    "\t[小妖精,fairy]在提升速度后,角色可以在更短的时间内打出足以击杀怪物的攻击频率。\n相比获取宝石之前,角色的攻击前摇快了25%。",
+                    "\t[小妖精,fairy]虽然攻击威力没有变化,仍然需要8次攻击才能击杀怪物,但战斗时间却实打实地从0.5降低到了0.4。\n在0.4时刻之前,怪物最多只能打出18*0.4=7次攻击。\n当然,提升攻击力降低战斗伤害的本质也是缩短战斗时间,从而让怪物的攻击频率缩短。",
+                    "\t[小妖精,fairy]但要注意——不是所有时候提升速度都能降低角色受到的攻击次数。\n如果角色的速度从21提升到24,击杀怪物需要的时间将由8/21=0.381变为0.333。(实际计算中具体时刻将保留14位小数,确保精确)\n怪物的第6次攻击于0.333时刻落下,而第7次攻击将于0.389时刻落下。\n因此,将击杀怪物的时刻从0.381提前到0.333,不影响怪物的第6和第7次攻击,从而不会降低伤害。",
+                    "\t[小妖精,fairy]因此,能带来战斗伤害减少的速度提升节点也特别重要——我们也称之为“临界”。\n从最初的需要8次攻击击杀,16点速度,0.5时刻击杀怪物开始计算(当然是计算怪物的时间轴)——\n怪物的第8次攻击落于0.444时刻,第9次攻击落于0.5时刻。\n速度达到17时,击杀时刻为8/17=0.471;速度达到19时,击杀时刻为8/19=0.421,刚好在怪物的第9次和第8次攻击之前。",
+                    {
+                        "type": "callBook"
+                    },
+                    "\t[小妖精,fairy]同理,以现在的状态开始计算。现在我们需要0.4的攻击时间击杀怪物。\n怪物最后一次的攻击时刻为0.389;我们提升1点速度将会使我们的最后一次攻击时刻提前到0.381,从而少受到一次怪物的攻击——这1点速度就是一个速度临界值。继续提升速度,就需要考虑怪物在0.333时刻的攻击。",
+                    "\t[小妖精,fairy]我们的速度距离下一个临界点还差1点;达到临界点可以让我们受到的伤害减少50点。\n在怪物手册中记作[1,50]或\"1:50\"。速度临界值和临界表暂时需要到怪物手册中才能查阅。",
+                    "\t[小妖精,fairy]捡起后面的血瓶和黄钥匙,补充生命值继续后面的内容吧!",
+                    {
+                        "type": "jump",
+                        "to": [
+                            5,
+                            2
+                        ],
+                        "time": 500,
+                        "keep": true
+                    }
+                ]
+            }
+        ],
+        "5,2": [
+            "\t[小妖精,fairy]防御是另一个非常重要的属性。\n与攻击和速度相比,防御是通过降低被怪物攻击的单次伤害来减少战斗损失的。\n对物理攻击的怪物来说,每一点防御都能起到应有的作用。",
+            "\t[小妖精,fairy]而且角色的防御与敌人的攻击越接近,防御的减伤效果越好。\n一般来说,如果只考虑攻击和防御的话,攻防和高于怪物的时候,防御的减伤效果比攻击更好。",
+            "\t[小妖精,fairy]在刚才的战斗中,怪物的攻击为28,角色的防御为16。怪物每次攻击对角色造成12点伤害。\n在获取黄门中的2颗蓝宝石后,角色的防御将从16提升到20,怪物的攻击将仅造成8点伤害。",
+            {
+                "type": "callBook"
+            },
+            "\t[小妖精,fairy]当然,1点防御的具体减伤价值,和战斗中角色受到的物理攻击次数有关,且怪物的特殊属性经常会造成影响。\n以刚才的战斗为例,角色总共被怪物攻击34次。\n1点防御将使怪物造成的普通攻击物理伤害减少1点,从而降低战斗伤害34点。\n同理,对于之前战斗过的两个怪物,1点防御对它们的价值分别是9点和7点减伤。",
+            "\t[小妖精,fairy]但对于初级法师来说——由于初级法师的攻击类型是魔法攻击,物理防御不能起到作用。\n因此尽管怪物出手次数为2,1点防御也无法减少任何伤害。",
+            {
+                "type": "jump",
+                "to": [
+                    4,
+                    9
+                ],
+                "time": 500,
+                "keep": true
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "4,9": [
+            {
+                "type": "if",
+                "condition": "(flag:跳过2!==1)",
+                "true": [
+                    "\t[小妖精,fairy]接下来要介绍的是最后一个主要战斗属性——法强。\n任何生灵都有属于自己的魔法能量——只是他们大多数不知道如何去使用。\n在本作的初始阶段,角色不了解魔法的运作方式,自身所有的魔法能量都在无意识地保护自己。\n1点魔法强度可以给角色带来1点护盾效果——每次战斗(而不是每回合)抵挡1点战斗伤害。\n这个护盾效果可以抵挡所有种类的伤害,不拘泥于物理,魔法或真实伤害。",
+                    "\t[小妖精,fairy]对于不懂如何使用魔法的怪物而言,其魔法能量被简单地合并到其生命值中进行计算。\n但对于会使用魔法攻击的怪物而言,其普通攻击或技能将造成魔法伤害。\n在怪物手册中,怪物普通攻击的魔法攻击力将显示在“法强”一栏。",
+                    {
+                        "type": "callBook"
+                    },
+                    "\t[小妖精,fairy]怪物造成的魔法伤害无法通过简单地提高攻击/防御/速度抵挡。\n但有些装备可以提供百分比的法抗(魔法伤害抗性)。\n携带多件百分比法抗装备的话,抗性乘法叠加计算。",
+                    "\t[小妖精,fairy]当然,在携带特定装备后,角色也有着使用魔法攻击的机会。\n不同类型的法杖将允许角色以不同比例将魔法能量向魔法攻击力转化。\n当然,因为角色将一部分魔力用于攻击,剩余支撑护盾的魔力就没那么多了。\n具体的魔法攻击力和护盾强度,将显示在状态栏对应比例的括号外面。\n括号里面的百分比是魔法能量以多大的比例转化为魔法攻击力和护盾。"
+                ]
+            },
+            {
+                "type": "choices",
+                "text": "\t[小妖精,fairy]现在,你已经熟悉了简单的战斗系统。\n接下来的进阶部分,还需要我的引导嘛?",
+                "choices": [
+                    {
+                        "text": "我太需要了",
+                        "action": [
+                            {
+                                "type": "jump",
+                                "to": [
+                                    2,
+                                    12
+                                ],
+                                "time": 500,
+                                "keep": true
+                            }
+                        ]
+                    },
+                    {
+                        "text": "这还是太简单了",
+                        "action": [
+                            {
+                                "type": "jump",
+                                "to": [
+                                    5,
+                                    0
+                                ],
+                                "time": 500
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        2,
+                                        12
+                                    ]
+                                ],
+                                "remove": true
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        6,
+                                        11
+                                    ]
+                                ],
+                                "floorId": "jiaocheng03",
+                                "remove": true
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        6,
+                                        9
+                                    ]
+                                ],
+                                "floorId": "jiaocheng03",
+                                "remove": true
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        6,
+                                        7
+                                    ]
+                                ],
+                                "floorId": "jiaocheng03",
+                                "remove": true
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        6,
+                                        4
+                                    ]
+                                ],
+                                "floorId": "jiaocheng03",
+                                "remove": true
+                            },
+                            {
+                                "type": "hide",
+                                "loc": [
+                                    [
+                                        6,
+                                        2
+                                    ]
+                                ],
+                                "floorId": "jiaocheng03",
+                                "remove": true
+                            }
+                        ]
+                    }
+                ]
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "2,12": [
+            "\t[小妖精,fairy]这里是一个非常简单的生命价值判断。\n首先,我们的生命和路上的血瓶不足以让我们携带2把蓝钥匙击败把守着绿色血瓶的怪物。\n其次,路上没有任何机会获得宝石,提高能力,来减少战斗伤害。\n因此,我们至少要打开一扇蓝门来保存血量。\n当然,考虑到后面需要使用蓝钥匙来换取战斗属性以击败更强大的敌人,这里只允许打开一扇蓝门。",
+            "\t[小妖精,fairy]我们一共有2个选择:打开中间的蓝门绕开怪物前进,或者击败一个怪物并打开蓝门获取250生命值的血瓶。",
+            {
+                "type": "callBook"
+            },
+            "\t[小妖精,fairy]首先,观察到所有怪物的金币、经验均为0点。这意味着不需要清理怪物获得资源。\n绕过挡路的怪物取得900血瓶的这扇蓝门,其生命价值为252-0=252点(为什么不是302点?)。\n而把守250生命值血瓶的蓝门,其生命价值为250点。\n显然,打开绕怪门的收益更大。打开我身后的这扇蓝门,继续后面的内容吧。\n如果在正式游戏过程中可以以0伤害清理怪物获得资源时,也适用这样的计算方法。",
+            {
+                "type": "jump",
+                "to": [
+                    5,
+                    0
+                ],
+                "time": 500
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "3,8": [
+            {
+                "type": "animationDrawable",
+                "allFarme": 30,
+                "color": [
+                    0,
+                    0,
+                    0
+                ],
+                "imageList": [
+                    {
+                        "image": ""
+                    }
+                ],
+                "soundList": [
+                    {
+                        "sound": "",
+                        "startfarme": 0,
+                        "stopbefore": false
+                    }
+                ]
+            }
+        ]
+    },
+    "changeFloor": {
+        "6,12": {
+            "floorId": "jiaocheng01",
+            "loc": [
+                6,
+                0
+            ]
+        },
+        "6,0": {
+            "floorId": "jiaocheng03",
+            "loc": [
+                6,
+                12
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  0, 31,  0,217, 33,  0, 91,  1,402,  1,404,  1,203],
+    [216,  1,  0,  1,  1,  1,  1,  1,402,  1,404,  1,  1],
+    [216,  1, 31,  1,  0,  0,  0, 81,  0,  1,  0,  1,425],
+    [425,  1, 21,  1,216,  1,216,  1,  1,  1, 81,  1,  1],
+    [  0,  1,  1,  1,  0,  1,  0,  0,  0,425,  0,  0,  0],
+    [ 31,  1, 32,  1,  0,  1,  1,  0,  1,  1,  1,  1,425],
+    [  0, 82,  0,  1,  0,  1, 21, 33,  1, 33,  0, 81,  0],
+    [  0,  1,  1,  1,217,  1,  1,  1,  1,  1,  1,  1, 22],
+    [203,  1, 33,  0,  0,403,403,403,403,  1,  0,  0,  0],
+    [  0,  1,  1,  1,  0,  1,  1,  1,  1,  1,203,  1, 82],
+    [ 33,  0,425,  1,  0,  1,  0,203,  0,124,  0,  1,  0],
+    [  0, 82,  0,  1,217,  1,  0,  1,  1, 81,  1,  1, 21],
+    [216,  0,  0, 22,  0,  1, 93,  1,401,  0,401,  1, 21]
+],
+    "areas": "教程",
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/jiaocheng03.js b/project/floors/jiaocheng03.js
new file mode 100644
index 0000000..756affb
--- /dev/null
+++ b/project/floors/jiaocheng03.js
@@ -0,0 +1,175 @@
+main.floors.jiaocheng03=
+{
+    "floorId": "jiaocheng03",
+    "title": "广场",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {
+        "0,0": [
+            "匕首:速度+20"
+        ],
+        "12,0": [
+            "法杖:魔攻比例10%,魔防比例0%"
+        ],
+        "6,0": [
+            {
+                "type": "setHeroOpacity",
+                "opacity": 0
+            },
+            {
+                "type": "function",
+                "function": "function(){\ncore.unloadEquip(0, () => {\n\tcore.setItem(\"sword0\", 0);\n\tcore.setItem(\"I375\", 0)\n\tcore.setItem(\"yellowKey\", 0)\n\tcore.setItem(\"blueKey\", 0)\n\tcore.setItem(\"redKey\", 0)\n\tcore.setItem(\"greenKey\", 0)\n\tcore.status.hero.hp = 1000\n\tcore.status.hero.atk = 16\n\tcore.status.hero.def = 16\n\tcore.status.hero.spell = 0\n\tcore.status.hero.speed = 20\n\tcore.updateStatusBar()\n})\n}"
+            },
+            {
+                "type": "tip",
+                "text": "教学层属性已清空"
+            },
+            {
+                "type": "changeFloor",
+                "floorId": "guangchang",
+                "loc": [
+                    6,
+                    10
+                ]
+            }
+        ],
+        "6,11": [
+            "\t[小妖精,fairy]接下来是能力价值判断。\n首先击败我右边的怪物,然后你需要在这三扇黄门中打开正确的一扇。\n打开错误的黄门将使你的生命值不足以抗住战斗伤害,并死亡。\n当然,你随时可以使用读取自动存档功能撤销错误的选择。",
+            {
+                "type": "jump",
+                "to": [
+                    6,
+                    9
+                ],
+                "time": 500,
+                "keep": true
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "6,9": [
+            "\t[小妖精,fairy]刚才你应该是选择了红宝石,对吗?\n选择蓝宝石将见不到我,而选择红血瓶你将会因为能力不够而折戟在后面的怪物上。\n在除决战时刻以外的场合,获取能力都应该优于获取生命值,哪怕是看似不起眼的法强/护盾。",
+            "\t[小妖精,fairy]刚才为什么要增加2点攻击而不是2点防御呢?\n你应该看过了怪物手册才对。\n攻击临界表中有一项\"2:84\",意味着2点攻击可以降低84点战斗伤害。\n而1防减伤仅有36,意味着2点防御仅能降低72点战斗伤害。\n多出来的这部分战斗伤害,暂时是不可承受之重。",
+            "\t[小妖精,fairy]实际战斗中又应该怎么规划呢?\n观察到,角色的攻击防御之和为50,而怪物的攻击防御之和为56。\n在攻击/防御二选一的时候,一般攻防和高于怪物时加防御效果更好,低于怪物时则选择爆攻击。\n当然,怪物生命极低,角色接近一刀将其秒杀时,加攻击的效果也很好。",
+            "\t[小妖精,fairy]接下来是含速度的能力价值判断。\n你需要在这三扇红门中打开正确的一扇。\n打开错误的红门将使你的生命值不足以抗住战斗伤害,并死亡。\n当然,你随时可以使用读取自动存档功能撤销错误的选择。",
+            {
+                "type": "jump",
+                "to": [
+                    6,
+                    7
+                ],
+                "time": 500,
+                "keep": true
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "6,7": [
+            "\t[小妖精,fairy]刚才你应该是选择了黄宝石,为什么呢?",
+            "\t[小妖精,fairy]首先排除2点防御力——敌人的单次攻击伤害高达93点,2点防御力的减伤杯水车薪。\n然后是在2点速度和2点攻击力中选择。\n在怪物手册中,速度临界值为1,可以减伤93点;攻击临界值为3,可以减伤93点。\n获取2点攻击力不足以达到怪物的攻击临界值。",
+            "\t[小妖精,fairy]实际战斗中又应该怎么决策呢?\n提高速度和攻击力都是为了减少被怪物攻击的次数,换句话说,缩短战斗的时间。\n因此,我们追求在更短的时间内打出更高的伤害,也就是更高的DPS。\n观察到,角色对怪物的DPS(速度*单次攻击伤害)为20*30=600。\n提高2点速度将使DPS提升为22*30=660,而提高2点攻击力将使DPS提升为20*32=640。\n简单地说,我们需要增加速度和(攻击-怪物防御)中较小的一项。就是个均值不等式。",
+            "\t[小妖精,fairy]下一个三选一见。\n怪物的“先攻”特殊属性,意味着怪物已经处于攻击姿态,在角色表现敌意的一霎那就会受到怪物的攻击。\n你需要在这三扇蓝门中打开正确的一扇。\n打开错误的蓝门将使你的生命值不足以抗住战斗伤害,并死亡。\n当然,你随时可以使用读取自动存档功能撤销错误的选择。",
+            {
+                "type": "jump",
+                "to": [
+                    6,
+                    4
+                ],
+                "time": 500,
+                "keep": true
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "6,4": [
+            "\t[小妖精,fairy]刚才你应该是选择了1黄宝石1红宝石的组合,但——到底是什么原理?",
+            "\t[小妖精,fairy]接下来要讲的是本作一个非常隐蔽的现象——混合临界。\n混合临界仅会出现在角色的速度高于怪物速度的时候。\n出现混合临界的条件是,在怪物对角色进行最后一次攻击后,角色进行了两次以上的攻击才击杀了怪物。\n为了方便理解,可以画一个时间轴,从怪物最后一次攻击算起。",
+            "\t[小妖精,fairy]单纯提高攻击力来降低被攻击次数的话,需要让角色击杀怪物的攻击次数减少2,而不是1。\n在刚才的战斗中,角色的攻击力为32,击杀怪物需要162/(32-1)=6次攻击。\n要达到降低被攻击次数效果的话,将攻击力提升至162/5+1=34点并不足够。\n需要将攻击力提升至162/4+1=42点。",
+            "\t[小妖精,fairy]同样地,单纯提高速度来降低被攻击次数的话,需要让角色的最后一次攻击提前到怪物攻击之前,而不是倒数第二次攻击。\n在刚才的战斗中,怪物最后一次攻击的时刻为3/14=0.214。\n角色在5/22=0.227,6/22=0.273两个时刻分别攻击了怪物,其中0.273时刻对怪物的攻击造成了击杀。\n要达到降低被攻击次数效果的话,将速度提升至5+/0.214=24点并不足够,因为角色需要6次攻击才能击杀怪物。\n需要将速度提升至6+/0.214=29点。",
+            "\t[小妖精,fairy]但值得庆幸的是,我们还要第三种办法——同时增强速度和攻击。\n我们少量提升攻击力至34点,使击杀怪物仅需要5次攻击。\n然后再少量提升速度至24点,使我们的第5次攻击落在0.214时刻,即怪物的最后一次攻击前。\n这样,我们仅用了4点能力提升,就达到了单独提升某项能力10点或7点的效果。",
+            "\t[小妖精,fairy]接下来是法强(护盾)和防御的价值判断。\n希望你能在这三扇黄门中打开正确的一扇。",
+            {
+                "type": "jump",
+                "to": [
+                    6,
+                    2
+                ],
+                "time": 500,
+                "keep": true
+            },
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ],
+        "6,2": [
+            "\t[小妖精,fairy]防御和护盾的比较就非常简单了。\n2点防御的价值是2*2=4点减伤,而5点法强的价值是5点减伤。\n如果仅以这一个怪物为准的话,获取5点法强是更优解。\n当然,我不会责备你刚才选择了红血瓶——因为没命的话,就啥也不是了。",
+            "\t[小妖精,fairy]但最终的守卫怪物是个坚固怪物——它每次攻击至多只会受到1点物理伤害。\n左边的红门里面,守着的是一把提高20速度的匕首。\n右边的红门里面,守着的是一把以10%效率将魔法强度转化为魔法攻击的法杖(当然护盾就没有了)。\n值得一提的是,本作魔法强度在转化成魔法攻击的时候,是向下取整的。",
+            "\t[小妖精,fairy]除此之外,你还拥有2把蓝钥匙来进一步提升自己的战斗力。\n为了击败面前的最终怪物,进行最后的二选一和七选二吧!\n不要忘了上一张地图内含250生命值的蓝门哦。\n综合利用攻防判断,DPS判断等手段,保留更多的生命值吧!",
+            {
+                "type": "hide",
+                "remove": true
+            }
+        ]
+    },
+    "changeFloor": {
+        "6,12": {
+            "floorId": "jiaocheng02",
+            "loc": [
+                6,
+                0
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [129,  1,401,  1,402,  1, 91,  1,403,  1,404,  1,129],
+    [ 70,  1, 82,  1, 82,  1,236,  1, 82,  1, 82,  1,375],
+    [  0, 83,  0,403,  0, 33,  0, 31,  0, 23,  0, 83,  0],
+    [  1,  1,  1,  1,  1,  1,219,  1,  1,  1,  1,  1,  1],
+    [  0, 22,  0, 31,  0, 31,  0,  1,401,  1,404,  1,404],
+    [ 81,  1, 81,  1, 81,  1, 22,  1,401,  1,404,  1,401],
+    [403,  1, 31,  1, 31,  1,218,  1, 82,  1, 82,  1, 82],
+    [402,  1,402,  1,403,  1,  0, 81,  0, 33,  0, 21,  0],
+    [  1,  1,  1,  1,  1,  1,240,  1,  1,  1,  1,  1,  1],
+    [  0, 33,  0, 21,  0,403,  0,  1,401,  1,402,  1, 31],
+    [ 83,  1, 83,  1, 83,  1,204,  1, 81,  1, 81,  1, 81],
+    [402,  1,404,  1,401,129,124,204,  0, 33,  0, 23,  0],
+    [  1,  1,  1,  1,  1,  1, 93,  1,  1,  1,  1,  1,  1]
+],
+    "areas": "牢狱",
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/jiuguan.js b/project/floors/jiuguan.js
index 02fa764..07f4309 100644
--- a/project/floors/jiuguan.js
+++ b/project/floors/jiuguan.js
@@ -32,6 +32,7 @@ main.floors.jiuguan=
                 "type": "over",
                 "text": "这段话将在屏幕中心",
                 "image": "bg_5043.png",
+                "memory": false,
                 "time": 3000,
                 "hidetime": 30,
                 "sound": "",
diff --git a/project/floors/street01.js b/project/floors/street01.js
index dceac7f..62ca09a 100644
--- a/project/floors/street01.js
+++ b/project/floors/street01.js
@@ -10,7 +10,7 @@ main.floors.street01=
     "canUseQuickShop": true,
     "images": [],
     "ratio": 1,
-    "defaultGround": "X90001",
+    "defaultGround": "X90078",
     "firstArrive": [],
     "eachArrive": [],
     "parallelDo": "",
@@ -25,68 +25,21 @@ main.floors.street01=
                 ],
                 "direction": "up"
             }
-        ],
-        "6,11": [
-            {
-                "type": "insert",
-                "name": "chapter04"
-            }
-        ],
-        "4,11": [
-            {
-                "type": "animationDrawable",
-                "allFarme": 300,
-                "color": [
-                    0,
-                    0,
-                    0
-                ],
-                "imageList": [
-                    {
-                        "image": "bg_1511.webp",
-                        "beforefarme": 50,
-                        "globalAlpha": 0,
-                        "afterfarme": 250,
-                        "aglobalAlpha": 100
-                    },
-                    {
-                        "image": "eve_010304.webp",
-                        "beforefarme": 70,
-                        "globalAlpha": 0,
-                        "afterfarme": 170,
-                        "aglobalAlpha": 100,
-                        "ax": 0,
-                        "ay": 0
-                    },
-                    {
-                        "image": "eve_010304.webp",
-                        "beforefarme": 171,
-                        "globalAlpha": 100,
-                        "afterfarme": 270,
-                        "aglobalAlpha": 100,
-                        "ax": 800,
-                        "ay": 0
-                    }
-                ],
-                "soundList": [
-                    {
-                        "sound": "",
-                        "startfarme": 0,
-                        "stopbefore": false
-                    }
-                ]
-            },
-            {
-                "type": "op"
-            }
         ]
     },
     "changeFloor": {
+        "12,9": {
+            "floorId": "zhujuejia",
+            "loc": [
+                0,
+                9
+            ]
+        },
         "0,8": {
-            "floorId": "street02",
+            "floorId": "changguan2",
             "loc": [
                 12,
-                6
+                8
             ]
         }
     },
@@ -116,27 +69,27 @@ main.floors.street01=
     [110124,  0,200176,200177,200178,200179,71786,90064,90064,90064,  0,90064,110125],
     [110124,  0,200088,200089,200004,200005,200006,71795,  0,  0,  0,  0,110125],
     [110118,110118,200096,110116,110020,  0,200014,  0,  0,  0,  0,  0,110125],
-    [ 92,  0,  0,110124,  0,201,  0,  0,110125,110122,110122,110122,110122],
-    [  0,  0,  0,110118,110118,202,110118,110118,110122,  0,  0,  0, 94],
-    [130170,  0,  0,  0, 71, 35,  0,  0,  0,  0,  0,  0,  0],
-    [130178,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,130168,130169],
-    [130225,130169,130169,130169,130169,130170,110122,110122,130168,130169,130169,130224,130177]
+    [ 92,  0,  0,110124,  0,  0,  0,  0,110125,110122,110122,110122,110122],
+    [  0,  0,  0,110118,110118,  0,110118,110118,110122,  0,  0,  0, 94],
+    [140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140,140],
+    [140,140,140,140,140,140186,110138,110138,140184,140,140,140,140]
 ],
     "areas": "牢狱",
     "bgmap": [
-    [90001,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90001],
-    [90001,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90001],
-    [90001,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90001],
-    [90001,71818,71819,  0,  0,  0,  0,  0,  0,  0,10036,  0,90001],
-    [90001,71826,71827,  0,  0,  0,  0,  0,  0,  0,  0,  0,90001],
-    [90001,  0,  0,  0,  0,  0,71786,  0,60135,60135,60135,60135,  0],
-    [90001,  0,  0,  0,  0,  0,  0,  0,  0,90001,90001,90001,  0],
-    [90001,90001,90001,  0,  0,  0,  0,  0,90001,  0,  0,  0,  0],
-    [70032,70032,70025,  0,  0,90244,90001,90001,90001,  0,  0,  0,  0],
-    [70025,70025,70025,90001,90001,90244,90001,90001,90001,70025,70025,70025,70025],
-    [  0,70025,70025,70025,70025,70025,70025,70025,70025,70025,70025,70025,70025],
-    [  0,70025,70025,70025,70025,70025,70025,70025,70025,70025,70025,  0,  0],
-    [  0,  0,  0,  0,  0,  0,90001,90001,  0,  0,  0,  0,  0]
+    [90078,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90078],
+    [90078,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90078],
+    [90078,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90078],
+    [90078,71818,71819,  0,  0,  0,  0,  0,  0,  0,10036,  0,90078],
+    [90078,71826,71827,  0,  0,  0,  0,  0,  0,  0,  0,  0,90078],
+    [90078,  0,  0,  0,  0,  0,71786,  0,60135,60135,60135,60135,90078],
+    [90078,  0,  0,  0,  0,  0,  0,  0,90078,90078,90078,90078,  0],
+    [90078,90078,90078,  0,  0,  0,  0,  0,90078,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,90244,90078,90078,90078,  0,  0,  0,  0],
+    [  0,  0,  0,90078,90078,90244,90078,90078,90078,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,90078,90078,  0,  0,  0,  0,  0]
 ],
     "fgmap": [
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,130076],
@@ -149,8 +102,8 @@ main.floors.street01=
     [  0,  0,  0,  0,  0,  0,  0,  0,110143,110114,110114,110114,110114],
     [  0,  0,  0,110114,120095,  0,120095,120095,110114,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,110114,110114,  0,  0,  0,  0,  0],
+    [  0,140186,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,140194,  0,  0,  0,  0,  0,  0,  0,  0,140184,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "bg2map": [
diff --git a/project/floors/street02.js b/project/floors/street02.js
index 2c49a82..9bce79d 100644
--- a/project/floors/street02.js
+++ b/project/floors/street02.js
@@ -16,13 +16,6 @@ main.floors.street02=
     "parallelDo": "",
     "events": {},
     "changeFloor": {
-        "12,6": {
-            "floorId": "street01",
-            "loc": [
-                0,
-                8
-            ]
-        },
         "0,6": {
             "floorId": "guangchang",
             "loc": [
@@ -30,6 +23,13 @@ main.floors.street02=
                 6
             ],
             "direction": "left"
+        },
+        "12,8": {
+            "floorId": "street01",
+            "loc": [
+                0,
+                8
+            ]
         }
     },
     "beforeBattle": {},
@@ -46,9 +46,9 @@ main.floors.street02=
     [141,141,141,141,141,141,141,141,141,141,141,141,141],
     [140,140,140,140,140,140,140,140,140,140,140,140,140],
     [  0,  0,140, 27,  0,210,  0,209,  0,  0,217,  0,  0],
-    [ 92,  0,140,140,140,140,  0,140,140,140,140,  0, 94],
+    [ 92,  0,140,140,140,140,  0,140,140,140,140,  0,140],
     [  0,  0,217,  0,  0,209,  0,210,  0, 28,140,  0,  0],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
+    [140,140,140,140,140,140,140,140,140,140,140,140, 94],
     [141,141,141,141,141,141,141,141,141,141,141,141,141],
     [141,141,141,141,141,141,141,141,141,141,141,141,141],
     [141,141,141,141,141,141,141,141,141,141,141,141,141],
diff --git a/project/floors/xiaoxiang01.js b/project/floors/xiaoxiang01.js
index 3943dfe..6cc4e72 100644
--- a/project/floors/xiaoxiang01.js
+++ b/project/floors/xiaoxiang01.js
@@ -16,13 +16,6 @@ main.floors.xiaoxiang01=
     "parallelDo": "",
     "events": {},
     "changeFloor": {
-        "0,10": {
-            "floorId": "guangchang2",
-            "loc": [
-                12,
-                10
-            ]
-        },
         "12,7": {
             "floorId": "xiaoxiang02",
             "loc": [
@@ -36,6 +29,13 @@ main.floors.xiaoxiang01=
                 6,
                 0
             ]
+        },
+        "0,7": {
+            "floorId": "guangchang2",
+            "loc": [
+                12,
+                10
+            ]
         }
     },
     "beforeBattle": {},
@@ -46,31 +46,79 @@ main.floors.xiaoxiang01=
     "cannotMove": {},
     "cannotMoveIn": {},
     "map": [
-    [141,141,141,141,141,141,141,141,141,141,141,141,141],
-    [141,141,141,141,141,141,141,141,141,141,141,141,141],
-    [141,141,141,141,141,141,141,141,141,141,141,141,141],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140, 30,140,  0, 81,  0,221,  0,224,  0, 28, 27,140],
-    [140,224,140,225,140, 31,140,  0,140,140,140,224,140],
-    [140,  0, 32,  0,140,  0,140,225,140,224,  0,  0,140],
-    [140,225,140,140,140, 81,140,  0,  0,  0,140,  0, 94],
-    [140,  0,222,  0,221,  0,140, 81,140,225,140,140,140],
-    [140,140,140, 81,140,140,140,  0,140,  0, 81, 31,140],
-    [ 92,  0, 32, 22, 32,140, 31,  0,140,  0,140,140,140],
-    [140, 21, 21, 21, 21,140,  0,  0, 81,  0,222, 29,140],
-    [140,140,140,140,140,140, 93,140,140,140,140,140,140]
+    [143,143,143,143,130193,130193,130193,130193,130193,143,143,143,143],
+    [143,143,143,143,130060,130061,130062,  0,120074,143,143,143,143],
+    [143,130193,130193,130194,130068,130069,130070,130090,120023,130192,130193,130193,143],
+    [130194,130201,130201,130202,130015,120041,130099,130098,120031,130200,130201,130201,143],
+    [130202, 30,201029,  0, 81,  0,221,  0,224,  0, 28, 27,130192],
+    [120139,224,201037,225,201037, 31,201021,  0,110121,110122,110122,224,130200],
+    [120147,  0, 32,  0,200199,  0,201029,225,110132,224,  0,  0,90675],
+    [120164,225,200998,200999,201037, 81,201029,  0,  0,  0,110116,  0, 94],
+    [120172,  0,222,  0,221,  0,201029, 81,110116,225,110137,110138,90650],
+    [120180,201037,201037, 81,201037,201037,201037,246,110124,  0, 81, 31,90658],
+    [ 92, 32, 32, 22, 21,80089, 31,  0,110132,  0,110122,110122,110197],
+    [140,110191, 21, 21, 21,80089,  0,246, 81,  0,222, 29,110197],
+    [140,140,140,110191,80083,80097, 93,110138,110138,140189,140,140,140]
 ],
     "areas": "牢狱",
     "bgmap": [
-
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
 ],
     "fgmap": [
-
+    [  0,  0,  0,  0,  0,120080,  0,  0,120066,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,130007,  0,130091,  0,  0,  0,  0,  0,  0],
+    [120124,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [120132,  0,  0,  0,201021,  0,  0,  0,110114,110114,110114,  0,  0],
+    [  0,  0,  0,  0,200191,  0,  0,  0,  0,  0,  0,  0,90667],
+    [  0,  0,200990,200991,201021,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,201021,201021,  0,201021,201021,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,110114,110114,110189],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "bg2map": [
-
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,130201,130201,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,120082,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,110134,110134,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "fg2map": [
-
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,201021,  0,200175,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,200183,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ]
 }
\ No newline at end of file
diff --git a/project/floors/xiaoxiang02.js b/project/floors/xiaoxiang02.js
index 13e9a9d..cd94f20 100644
--- a/project/floors/xiaoxiang02.js
+++ b/project/floors/xiaoxiang02.js
@@ -32,29 +32,65 @@ main.floors.xiaoxiang02=
     "cannotMove": {},
     "cannotMoveIn": {},
     "map": [
-    [141,141,141,141,141,141,141,141,141,141,141,141,141],
-    [141,141,141,141,141,141,141,141,141,141,141,141,141],
-    [141,141,141,141,141,141,141,141,141,141,141,141,141],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,  0, 81,  0,140,  0,  0,  0,  0,140,  0,  0,140],
-    [140,  0,140,  0,140,140,140,140, 81,140,140, 81,140],
-    [140,  0,140,  0,  0,  0,  0,  0,  0,  0,  0,  0,140],
-    [ 92,  0,140,140,140,140, 81,140,140,140, 81,140,140],
-    [140,  0,  0,  0,  0,  0,  0,  0,  0,140,  0,  0,140],
-    [140,140,140,140,140,  0,140,140, 82,140,140,  0,140],
-    [140,  0,  0,  0,140,  0,140,  0,  0,  0, 81,  0,140],
-    [140,  0,  0,  0, 81,  0,140,  0,  0,  0,140,  0,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140]
+    [143,143,143,143,143,120166,120166,110205,110206,110206,110206,110206,110206],
+    [143,143,143,143,143,120174,120174,130192,130193,130193,130193,130193,130193],
+    [143,130193,130193,130193,130194,190142,90771,130192,130193,130193,130193,130193,130193],
+    [143,130201,130201,130201,130202,190150,190150,130200,130201,  0,130201,130201,130201],
+    [130194,  0, 81,  0,72106,  0,  0,  0,  0,72106,  0,  0,143],
+    [130202,  0,110116,  0,90064,90064,90064,90064, 81,90064,90064, 81,143],
+    [90675,  0,110124,  0,  0,  0,  0,  0,  0,  0,  0,  0,143],
+    [ 92,  0,110122,110122,110122,110122, 81,110122,110122,110122, 81,110122,143],
+    [90651,  0,  0,  0,  0,  0,  0,  0,  0,110129,  0,  0,143],
+    [90659,71844,110122,200199,110122,  0,110121,110122, 82,110122,110122,  0,143],
+    [110199,  0,  0,  0, 81,  0,110129,  0,  0,  0, 81,  0,130192],
+    [110199,  0,  0,  0,110117,  0,110129,  0,  0,  0,200637,  0,143],
+    [140,140,140,140,110191,72098,72098,72098,143,143,143,143,143]
 ],
     "areas": "牢狱",
     "bgmap": [
-
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
 ],
     "fgmap": [
-
+    [  0,  0,  0,  0,  0,200063,200063,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90799,110072],
+    [  0,  0,90804,  0,  0,190023,190148,  0,90797,  0,  0,  0,110080],
+    [  0,  0,  0,  0,72112,  0,190030,  0,90805,72112,  0,  0,110088],
+    [  0,  0,  0,  0,  0,72098,72098,72098,  0,  0,72098,  0,  0],
+    [90667,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,110114,110114,110114,110114,  0,110114,110114,110114,  0,110114,  0],
+    [  0,  0,  0,200175,  0,  0,  0,  0,  0,110116,  0,  0,  0],
+    [  0,110114,110114,200191,110114,  0,110114,110114,  0,110114,110114,  0,  0],
+    [110191,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,200638,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "bg2map": [
-
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,110122,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,110133,  0,110132,  0,  0,  0,  0,  0,  0]
 ],
     "fg2map": [
 
diff --git a/project/floors/xiaoxiang3.js b/project/floors/xiaoxiang3.js
index 9b9d310..06871b2 100644
--- a/project/floors/xiaoxiang3.js
+++ b/project/floors/xiaoxiang3.js
@@ -14,7 +14,68 @@ main.floors.xiaoxiang3=
     "firstArrive": [],
     "eachArrive": [],
     "parallelDo": "",
-    "events": {},
+    "events": {
+        "4,8": {
+            "trigger": null,
+            "enable": false,
+            "noPass": null,
+            "displayDamage": true,
+            "opacity": 1,
+            "filter": {
+                "blur": 0,
+                "hue": 0,
+                "grayscale": 0,
+                "invert": false,
+                "shadow": 0
+            },
+            "data": []
+        },
+        "8,8": {
+            "trigger": null,
+            "enable": false,
+            "noPass": null,
+            "displayDamage": true,
+            "opacity": 1,
+            "filter": {
+                "blur": 0,
+                "hue": 0,
+                "grayscale": 0,
+                "invert": false,
+                "shadow": 0
+            },
+            "data": []
+        },
+        "7,10": {
+            "trigger": null,
+            "enable": false,
+            "noPass": null,
+            "displayDamage": true,
+            "opacity": 1,
+            "filter": {
+                "blur": 0,
+                "hue": 0,
+                "grayscale": 0,
+                "invert": false,
+                "shadow": 0
+            },
+            "data": []
+        },
+        "5,10": {
+            "trigger": null,
+            "enable": false,
+            "noPass": null,
+            "displayDamage": true,
+            "opacity": 1,
+            "filter": {
+                "blur": 0,
+                "hue": 0,
+                "grayscale": 0,
+                "invert": false,
+                "shadow": 0
+            },
+            "data": []
+        }
+    },
     "changeFloor": {
         "6,0": {
             "floorId": "xiaoxiang01",
@@ -28,117 +89,112 @@ main.floors.xiaoxiang3=
     "afterBattle": {
         "6,8": [
             {
-                "type": "setValue",
-                "name": "flag:door_xiaoxiang3_4_9",
-                "operator": "+=",
-                "value": "1"
-            },
-            {
-                "type": "setValue",
-                "name": "flag:door_xiaoxiang3_8_9",
-                "operator": "+=",
-                "value": "1"
-            },
-            {
-                "type": "setValue",
-                "name": "flag:door_xiaoxiang3_6_11",
-                "operator": "+=",
-                "value": "1"
+                "type": "show",
+                "loc": [
+                    [
+                        4,
+                        8
+                    ],
+                    [
+                        5,
+                        10
+                    ],
+                    [
+                        7,
+                        10
+                    ],
+                    [
+                        8,
+                        8
+                    ],
+                    [
+                        6,
+                        12
+                    ]
+                ],
+                "time": 500
             }
         ]
     },
     "afterGetItem": {},
     "afterOpenDoor": {},
-    "autoEvent": {
-        "4,9": {
-            "0": {
-                "condition": "flag:door_xiaoxiang3_4_9==1",
-                "currentFloor": true,
-                "priority": 0,
-                "delayExecute": false,
-                "multiExecute": false,
-                "data": [
-                    {
-                        "type": "openDoor"
-                    },
-                    {
-                        "type": "setValue",
-                        "name": "flag:door_xiaoxiang3_4_9",
-                        "operator": "=",
-                        "value": "null"
-                    }
-                ]
-            }
-        },
-        "8,9": {
-            "0": {
-                "condition": "flag:door_xiaoxiang3_8_9==1",
-                "currentFloor": true,
-                "priority": 0,
-                "delayExecute": false,
-                "multiExecute": false,
-                "data": [
-                    {
-                        "type": "openDoor"
-                    },
-                    {
-                        "type": "setValue",
-                        "name": "flag:door_xiaoxiang3_8_9",
-                        "operator": "=",
-                        "value": "null"
-                    }
-                ]
-            }
-        },
-        "6,11": {
-            "0": {
-                "condition": "flag:door_xiaoxiang3_6_11==1",
-                "currentFloor": true,
-                "priority": 0,
-                "delayExecute": false,
-                "multiExecute": false,
-                "data": [
-                    {
-                        "type": "openDoor"
-                    },
-                    {
-                        "type": "setValue",
-                        "name": "flag:door_xiaoxiang3_6_11",
-                        "operator": "=",
-                        "value": "null"
-                    }
-                ]
-            }
-        }
-    },
+    "autoEvent": {},
     "cannotMove": {},
     "cannotMoveIn": {},
     "map": [
-    [140,140,140,140,  0,  0, 91,  0,  0,140,140,140,140],
-    [140,140,140,140,  0,  0,  0,  0,  0,140,140,140,140],
-    [140,140,140,140,140,140,  0,140,140,140,140,140,140],
-    [140,140,140,140,140,140,  0,140,  0,140,140,140,140],
-    [140,140,140,140,  0,  0,  0,  0,  0,140,140,140,140],
-    [140,140,140,140,140,140, 83,140,140,140,140,140,140],
-    [140,140,140,140,  0,  0,  0,  0,  0,140,140,140,140],
-    [140,140,140,140,  0,  0,  0,  0,  0,140,140,140,140],
-    [140,140,140,140,  0,  0,247,  0,  0,140,140,140,140],
-    [140,140,140,140, 85,140,  0,140, 85,140,140,140,140],
-    [140,140,140,140, 34,140,  0,140, 27,140,140,140,140],
-    [140,140,140,140, 21,140, 85,140, 28,140,140,140,140],
-    [140,140,140,140,140,140, 93,140,140,140,140,140,140]
+    [  0,  0,  0,140199,80091,80092, 91,110146,110146,140197,  0,  0,  0],
+    [  0,  0,  0,140,140,140,  0,140,  0,  0,  0,  0,  0],
+    [  0,  0,  0,140204,140206,140206,140206,140206,140206,140203,  0,  0,  0],
+    [  0,  0,  0,140199,170263,170252,170253,170254,170263,140197,  0,  0,  0],
+    [  0,  0,  0,140199,170263,170260,  0,170262,170263,140197,  0,  0,  0],
+    [  0,  0,  0,140199,170263,170268, 83,170270,170263,140205,140206,140206,140206],
+    [  0,  0,  0,140199,71843,110122,  0,110122,90649,110208,110209,110209,110209],
+    [140206,140206,140206,140207,  0,  0,  0,  0,  0,110189,140,140,140],
+    [110209,110209,110209,110210, 21,  0,247,  0, 27,110197,140,140,140],
+    [140,140,140,110191,  0,  0,  0,  0,  0,110197,140,140,140],
+    [140,140,140,110199,  0, 34,  0, 28,  0,140,140,140,140],
+    [140,140,140,110199,110199,  0,  0,  0,110197,110204,110206,110206,110206],
+    [140,140,140,110199,110207,  0,  0,  0,110197,110199,110225,110225,110225]
 ],
     "areas": "牢狱",
     "bgmap": [
-
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
 ],
     "fgmap": [
-
+    [140,140,140,  0,  0,  0,  0,  0,  0,  0,140,140,140],
+    [140,140,140,140,140,140,140,140,140,140,140,140,140],
+    [140,140,140,  0,  0,  0,140206,110327,  0,  0,140,140,140],
+    [140,140,140,  0,  0,  0,170253,110335,  0,  0,140,140,140],
+    [140,140,140,  0,  0,110153,  0,110152,  0,  0,140,140,140],
+    [140,140,140,  0,71835,110161,  0,110160,90648,  0,  0,  0,  0],
+    [140,140,140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,110191,  0,  0,  0,110189,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,110138,  0,110138,  0,  0,  0,  0,  0]
 ],
     "bg2map": [
-
+    [140,140,140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140,140,140,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,110216,  0,  0,  0],
+    [  0,  0,  0,110218,  0,  0,  0,  0,  0,110216,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,110218,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "fg2map": [
-
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,110319,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,110114,  0,110114,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ]
 }
\ No newline at end of file
diff --git a/project/floors/yiqu1.js b/project/floors/yiqu1.js
index 0fbd450..195b38f 100644
--- a/project/floors/yiqu1.js
+++ b/project/floors/yiqu1.js
@@ -11,41 +11,119 @@ main.floors.yiqu1=
     "images": [],
     "ratio": 1,
     "defaultGround": "grass2",
-    "firstArrive": [],
+    "firstArrive": [
+        {
+            "type": "setHeroOpacity",
+            "opacity": 1
+        },
+        {
+            "type": "setCurtain",
+            "time": 100
+        }
+    ],
     "eachArrive": [],
     "parallelDo": "",
     "events": {},
-    "changeFloor": {},
+    "changeFloor": {
+        "12,8": {
+            "floorId": "yiqu2",
+            "loc": [
+                0,
+                8
+            ]
+        },
+        "0,7": {
+            "floorId": "guangchang6",
+            "loc": [
+                12,
+                7
+            ]
+        }
+    },
     "beforeBattle": {},
     "afterBattle": {},
     "afterGetItem": {},
     "afterOpenDoor": {},
     "autoEvent": {},
-    "cannotMove": {},
-    "cannotMoveIn": {},
+    "cannotMove": {
+        "8,1": [
+            "right"
+        ]
+    },
+    "cannotMoveIn": {
+        "8,1": [
+            "right"
+        ]
+    },
     "map": [
-    [140,140,140,140,140,140, 91,140,140,140,140,140,140],
-    [140,140,140,140,140,140,  0,140,140,  0,  0,  0, 94],
-    [140,140,140,140,140,140,  0,  0,140,140, 81,  0,  0],
-    [140,140,140,140,140,140,  0,  0,  0,140,  0,140,140],
-    [140,  0,  0,  0,140,  0,  0,  0,140,140,140,140,140],
-    [140,  0,140,140,140,  0,140,  0,140,  0,  0,  0, 94],
-    [140,  0,  0,140,  0,  0,140,  0,  0,  0,  0,  0,140],
-    [ 92,  0,  0,140,140,140,140,  0,140,140, 81,140,140],
-    [140,140,  0,140,  0,  0,  0,  0,140,  0,  0,  0,140],
-    [140,  0,  0,  0,  0,  0,140,  0,140,  0,  0,  0,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140]
+    [143,143,143,143,130193,130193,130193,130193,130193,143,143,143,143],
+    [143,143,143,143,130060,130061,130062,130201,130201,143,143,143,143],
+    [143,130193,130193,130194,130068,130069,130070,  0,120023,130192,130193,130193,143],
+    [130194,130201,130201,130202,130007,130006,130099,130098,120031,130200,130201,130201,143],
+    [130202, 32,  0, 21,201021,401,110116,  0,201, 81,  0, 21,130192],
+    [140101,  0,205,  0,201037,421,110124, 31,  0,110116, 31,  0,130200],
+    [140109,201037,  0,201037,200199, 81,110140,110138, 81,110138,110138,202,90667],
+    [ 92,201,  0,  0, 31,  0,202,  0,  0, 82,  0,  0,90675],
+    [120111,  0,  0, 21,  0,  0,201021,205,110138,110138,201021,  0, 94],
+    [120119,110122, 81,110122,201037,201,201029,402,  0, 21,201029,205,90650],
+    [120127,202, 31,202,201037,  0,201029,  0, 21,  0,201029,  0,110189],
+    [120135, 22,205,403,201037,202,201037,110122,110122, 81,201037,201,110197],
+    [143,130193,140170,130194,130068,130069,130070,  0,120023,130192,130157,130193,143],
+    [130194,130201,140178,130202,130007,130006,130099,130098,120031,130200,130167,130201,143],
+    [130202, 32,  0, 21,201021,401,220008,  0,201, 81,  0, 21,130192],
+    [140101,  0,205,  0,201037,421,220008, 31,  0,220008, 31,  0,130200],
+    [140109,201037,  0,201037,200199, 81,220012,220013, 81,220012,220013,202,90667],
+    [ 92,201,  0, 21,  0,  0,202,  0, 21,  0, 82,  0,90675],
+    [120111,  0,  0,  0,  0,  0,201021,205,110138,110138,201021,  0, 94],
+    [120119,220013, 81,220013,201037,201,201029,  0, 31,  0,201029,205,90650],
+    [120127,202, 31,202,201037,  0,201029, 21,  0,402,201029,  0,110189],
+    [120135, 22,205,403,201037,202,201037,220013,220013, 81,201037,201,110197],
+    [120143,110138,110138,110138,201029,  0,201,  0, 31,  0, 21,  0,110197]
 ],
     "bgmap": [
-
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
 ],
     "fgmap": [
-
+    [  0,  0,  0,  0,  0,120080,  0,  0,120066,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,120074,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,130091,130090,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,200175,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140093,  0,  0,  0,200183,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,201021,  0,201021,200191,  0,  0,220009,  0,  0,220009,  0,90667],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,220009,  0,220009,201021,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,201021,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,201021,  0,  0,220009,220009,  0,  0,  0,  0],
+    [  0,  0,  0,  0,201021,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "bg2map": [
-
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,130193,  0,  0,  0,  0,  0,120082,  0,130193,  0,  0],
+    [  0,  0,130201,  0,  0,  0,  0,  0,  0,  0,130201,  0,  0],
+    [  0,  0,  0,  0,130015,  0,  0,  0,  0,  0,  0,  0,  0],
+    [130076,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [130084,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,110146,110146,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90658],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "fg2map": [
 
diff --git a/project/floors/yiqu10.js b/project/floors/yiqu10.js
index 98f1172..b276418 100644
--- a/project/floors/yiqu10.js
+++ b/project/floors/yiqu10.js
@@ -1,29 +1,29 @@
 main.floors.yiqu10=
 {
-"floorId": "yiqu10",
-"title": "主塔 10 层",
-"name": "10",
-"width": 13,
-"height": 13,
-"canFlyTo": true,
-"canFlyFrom": true,
-"canUseQuickShop": true,
-"images": [],
-"ratio": 1,
-"defaultGround": "grass2",
-"firstArrive": [],
-"eachArrive": [],
-"parallelDo": "",
-"events": {},
-"changeFloor": {},
-"beforeBattle": {},
-"afterBattle": {},
-"afterGetItem": {},
-"afterOpenDoor": {},
-"autoEvent": {},
-"cannotMove": {},
-"cannotMoveIn": {},
-"map": [
+    "floorId": "yiqu10",
+    "title": "主塔 10 层",
+    "name": "10",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {},
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
@@ -38,4 +38,16 @@ main.floors.yiqu10=
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
 }
\ No newline at end of file
diff --git a/project/floors/yiqu2.js b/project/floors/yiqu2.js
index 7dca682..9ebbfdc 100644
--- a/project/floors/yiqu2.js
+++ b/project/floors/yiqu2.js
@@ -10,12 +10,27 @@ main.floors.yiqu2=
     "canUseQuickShop": true,
     "images": [],
     "ratio": 1,
-    "defaultGround": "grass2",
+    "defaultGround": "X90078",
     "firstArrive": [],
     "eachArrive": [],
     "parallelDo": "",
     "events": {},
-    "changeFloor": {},
+    "changeFloor": {
+        "12,7": {
+            "floorId": "yiqu3",
+            "loc": [
+                0,
+                7
+            ]
+        },
+        "0,8": {
+            "floorId": "yiqu1",
+            "loc": [
+                12,
+                8
+            ]
+        }
+    },
     "beforeBattle": {},
     "afterBattle": {},
     "afterGetItem": {},
@@ -24,28 +39,64 @@ main.floors.yiqu2=
     "cannotMove": {},
     "cannotMoveIn": {},
     "map": [
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,140,140,140,140,140,140,140, 32, 21, 28, 27,140],
-    [140,140,140,140,140,140,140,140,140, 82,140,140,140],
-    [140, 32,  0,  0,210,  0,140,  0,  0,214,  0,  0,140],
-    [140,140,140, 81,140,  0, 81,  0,140, 21,140,  0,140],
-    [140,  0,209,  0,140,140,140,210,140,140,140,  0,140],
-    [140,210,140,214,140, 27,140,  0,140, 22,  0,  0, 94],
-    [140,  0,209,  0, 81, 28,140,  0, 81, 21, 21, 21,  0],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140],
-    [140,140,140,140,140,140,140,140,140,140,140,140,140]
+    [143,120034,140120,140121,120034,120012,  1,  1,  1,  1,  1,  1,  1],
+    [143,170007,140128,140129,170007,120019,  1,  1,  1,  1,  1,  1,  1],
+    [143,  0,  0,  0,  0,120019,  1,  1,  1,  1,  1,  1,  1],
+    [143, 81,  0, 22,  0,120020,  1,  1,  1,  1,  1,  0, 22],
+    [130194, 81,120014,120010,120010,120028,  1,  1,  1,  0, 83,  0,403],
+    [130202, 31,  0,209,  0,402,  1,  0, 34,  0,  1,209,  0],
+    [401,  0,202,  1, 32,  0,  1,405,  0,408,  1, 81,  1],
+    [  1,  1, 81,  1,  0,421,  1,  0,204,  0,  1,  0, 94],
+    [ 92,  0, 21,  1,  1, 81,  1,  1, 82,  1,  1,205,  1],
+    [  1,  1,  0,202,  0,  0, 32,209,  0,  1, 21,  0,  1],
+    [  0,201,  0,  1, 81,  1,  1,  1,  0,205,  0,  0,421],
+    [202,  1,  1,  1,  0,  1,402,  1,201,  1, 31,  1,  0],
+    [  0, 21, 21,  1,  0,202,  0,201,  0,  1,  1,  1, 22]
 ],
     "bgmap": [
-
+    [90078,90078,180033,180033,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
 ],
     "fgmap": [
-
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,120002,120002,120002,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "bg2map": [
-
+    [  0,180025,120034,120034,180025,180026,  0,  0,  0,  0,  0,  0,  0],
+    [  0,180025,180025,180025,180025,180026,  0,  0,  0,  0,  0,  0,  0],
+    [  0,180025,180025,180025,180025,180026,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,180025,130061,130060,180026,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
     "fg2map": [
 
diff --git a/project/floors/yiqu3.js b/project/floors/yiqu3.js
index 394234a..6acf6a5 100644
--- a/project/floors/yiqu3.js
+++ b/project/floors/yiqu3.js
@@ -1,41 +1,68 @@
 main.floors.yiqu3=
 {
-"floorId": "yiqu3",
-"title": "主塔 3 层",
-"name": "3",
-"width": 13,
-"height": 13,
-"canFlyTo": true,
-"canFlyFrom": true,
-"canUseQuickShop": true,
-"images": [],
-"ratio": 1,
-"defaultGround": "grass2",
-"firstArrive": [],
-"eachArrive": [],
-"parallelDo": "",
-"events": {},
-"changeFloor": {},
-"beforeBattle": {},
-"afterBattle": {},
-"afterGetItem": {},
-"afterOpenDoor": {},
-"autoEvent": {},
-"cannotMove": {},
-"cannotMoveIn": {},
-"map": [
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+    "floorId": "yiqu3",
+    "title": "主塔 3 层",
+    "name": "3",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "0,7": {
+            "floorId": "yiqu2",
+            "loc": [
+                12,
+                7
+            ]
+        },
+        "5,0": {
+            "floorId": "yiqu4",
+            "loc": [
+                5,
+                12
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  1,  1,  1,  1,  1, 91,  1,  1,  1,  1,  1,  1,  1],
+    [  1,  1,  1,  1,  1,  0,  1,  1,  1, 31, 21,  1,  1],
+    [  1,  1,  1,  1,  1,203,  1,  1,  1,  1, 21,  1,  1],
+    [  1,  1,  1,  1,  1,  0,  1,  1,  1,  1,  0,  1,  1],
+    [  1,  1,  1,  1,  1,205,  1,  1,  1,  1, 82,  1,  1],
+    [  1,  0,202,  0,421,  0,421,  0, 31,  0,  0,  1,  1],
+    [  1,201,  1,  1, 81,  1,  1,  1,  1,  1, 81,  1,  1],
+    [ 92,  0,  1, 31,  0,209,  0,  1,403,  0,209,  1,  1],
+    [  1, 81,  1,  0,244,  0, 31,  1,  0,403,  0,  1, 23],
+    [205,  0,  1,  1,  1,  1, 82,  1,205,  1,  1,  1,  0],
+    [  0,209,  1,401,402,  0,  0,  1,  0,  1,244,  0, 31],
+    [205,  0,  1,  1,  1,  1,216,  1, 81,  1,  0, 81,  0],
+    [ 21,  0,203,  0, 31, 21,  0, 81,  0, 82,216,  0,203]
 ],
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
 }
\ No newline at end of file
diff --git a/project/floors/yiqu4.js b/project/floors/yiqu4.js
index 498b6cb..7462679 100644
--- a/project/floors/yiqu4.js
+++ b/project/floors/yiqu4.js
@@ -1,41 +1,68 @@
 main.floors.yiqu4=
 {
-"floorId": "yiqu4",
-"title": "主塔 4 层",
-"name": "4",
-"width": 13,
-"height": 13,
-"canFlyTo": true,
-"canFlyFrom": true,
-"canUseQuickShop": true,
-"images": [],
-"ratio": 1,
-"defaultGround": "grass2",
-"firstArrive": [],
-"eachArrive": [],
-"parallelDo": "",
-"events": {},
-"changeFloor": {},
-"beforeBattle": {},
-"afterBattle": {},
-"afterGetItem": {},
-"afterOpenDoor": {},
-"autoEvent": {},
-"cannotMove": {},
-"cannotMoveIn": {},
-"map": [
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+    "floorId": "yiqu4",
+    "title": "主塔 4 层",
+    "name": "4",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "5,12": {
+            "floorId": "yiqu3",
+            "loc": [
+                5,
+                0
+            ]
+        },
+        "12,11": {
+            "floorId": "yiqu5",
+            "loc": [
+                0,
+                11
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1],
+    [  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1],
+    [  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1],
+    [  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1],
+    [  1,  1,  1,421,  0,205,  0,421,  1,  1,  1,  1,  1],
+    [  1,  1,  1,  0,  1, 81,  1,  0, 81, 81,  0,205,  0],
+    [  0,244,  0, 31,  1,  0,  1, 31,  1,  1,216,  1, 31],
+    [  0,  1,425,  1,  1, 31,  1,  1, 22,  1,  0,  1,  1],
+    [ 31,  1,  0,  1,401,  1,402,  1,  0,  1, 32,  1,403],
+    [  0, 81, 21,  1,  0,  1,  0,  1,244, 81,  0, 81,  0],
+    [403,  1,  0,  1, 82,  1, 82,  1,  1,  1,216,  1,  1],
+    [  0,  1,421,  0, 21,  0, 32,  0,240, 21,  0,425, 94],
+    [  0,216,  0,  1,  1, 93,  1,  1,  1,  1,  1,  1,  1]
 ],
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
 }
\ No newline at end of file
diff --git a/project/floors/yiqu5.js b/project/floors/yiqu5.js
index 74f5bdd..2ebfddd 100644
--- a/project/floors/yiqu5.js
+++ b/project/floors/yiqu5.js
@@ -1,41 +1,61 @@
 main.floors.yiqu5=
 {
-"floorId": "yiqu5",
-"title": "主塔 5 层",
-"name": "5",
-"width": 13,
-"height": 13,
-"canFlyTo": true,
-"canFlyFrom": true,
-"canUseQuickShop": true,
-"images": [],
-"ratio": 1,
-"defaultGround": "grass2",
-"firstArrive": [],
-"eachArrive": [],
-"parallelDo": "",
-"events": {},
-"changeFloor": {},
-"beforeBattle": {},
-"afterBattle": {},
-"afterGetItem": {},
-"afterOpenDoor": {},
-"autoEvent": {},
-"cannotMove": {},
-"cannotMoveIn": {},
-"map": [
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
-    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+    "floorId": "yiqu5",
+    "title": "主塔 5 层",
+    "name": "5",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "0,11": {
+            "floorId": "yiqu4",
+            "loc": [
+                12,
+                11
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1,  1],
+    [  1,  1,  1,  1,  1,  1,  0,  1,  1,  1,  1,  1,  1],
+    [  1,  0, 32,  1,  1,  1,426,  1,  1,  1,  1,  1,  1],
+    [  1, 82,  1,  1,  1,  1,  0,  1,  1, 31,  1,  1,  1],
+    [216,  0,216,  1,  1,  1, 83,  1,  1,  0,  1,  1,  1],
+    [  0,  1,  0,  1,  1,209,  0,209,  1, 81,  1,  1,  1],
+    [ 32,  1, 32,  0, 81,  0, 82,  0, 32,  0,216,  0, 22],
+    [  1,  1,  0,  1,  1,244,  1,  1, 81,  1,  1,  1,  1],
+    [ 31,  1,209,  1, 31,  0,403,  1,203,  0,244,  1, 31],
+    [  0,216,  0,  1,  1,  1,  1,  1,  0,203,  0,  1,  0],
+    [  1,  1, 32, 82,  0,402,  1,  1,209,  0, 34,  1, 81],
+    [ 92,  0,  0,  1,  1,  1,  1, 21,  1,  1, 21,  1, 22],
+    [  1,  1, 31, 82,  0,401,  1, 32,  0,425,  0,213,  0]
 ],
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
 }
\ No newline at end of file
diff --git a/project/floors/yiqu6.js b/project/floors/yiqu6.js
index 1399cff..633e6a3 100644
--- a/project/floors/yiqu6.js
+++ b/project/floors/yiqu6.js
@@ -1,29 +1,29 @@
 main.floors.yiqu6=
 {
-"floorId": "yiqu6",
-"title": "主塔 6 层",
-"name": "6",
-"width": 13,
-"height": 13,
-"canFlyTo": true,
-"canFlyFrom": true,
-"canUseQuickShop": true,
-"images": [],
-"ratio": 1,
-"defaultGround": "grass2",
-"firstArrive": [],
-"eachArrive": [],
-"parallelDo": "",
-"events": {},
-"changeFloor": {},
-"beforeBattle": {},
-"afterBattle": {},
-"afterGetItem": {},
-"afterOpenDoor": {},
-"autoEvent": {},
-"cannotMove": {},
-"cannotMoveIn": {},
-"map": [
+    "floorId": "yiqu6",
+    "title": "主塔 6 层",
+    "name": "6",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {},
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
@@ -38,4 +38,16 @@ main.floors.yiqu6=
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
 }
\ No newline at end of file
diff --git a/project/floors/yiqu7.js b/project/floors/yiqu7.js
index d47401f..459aded 100644
--- a/project/floors/yiqu7.js
+++ b/project/floors/yiqu7.js
@@ -1,29 +1,29 @@
 main.floors.yiqu7=
 {
-"floorId": "yiqu7",
-"title": "主塔 7 层",
-"name": "7",
-"width": 13,
-"height": 13,
-"canFlyTo": true,
-"canFlyFrom": true,
-"canUseQuickShop": true,
-"images": [],
-"ratio": 1,
-"defaultGround": "grass2",
-"firstArrive": [],
-"eachArrive": [],
-"parallelDo": "",
-"events": {},
-"changeFloor": {},
-"beforeBattle": {},
-"afterBattle": {},
-"afterGetItem": {},
-"afterOpenDoor": {},
-"autoEvent": {},
-"cannotMove": {},
-"cannotMoveIn": {},
-"map": [
+    "floorId": "yiqu7",
+    "title": "主塔 7 层",
+    "name": "7",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {},
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
@@ -38,4 +38,16 @@ main.floors.yiqu7=
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
 }
\ No newline at end of file
diff --git a/project/floors/yiqu8.js b/project/floors/yiqu8.js
index 18bb1e8..71c2a6f 100644
--- a/project/floors/yiqu8.js
+++ b/project/floors/yiqu8.js
@@ -10,7 +10,7 @@ main.floors.yiqu8=
     "canUseQuickShop": true,
     "images": [],
     "ratio": 1,
-    "defaultGround": "grass2",
+    "defaultGround": "X90078",
     "firstArrive": [],
     "eachArrive": [],
     "parallelDo": "",
diff --git a/project/floors/yiqu9.js b/project/floors/yiqu9.js
index 2b2d3e7..0bab892 100644
--- a/project/floors/yiqu9.js
+++ b/project/floors/yiqu9.js
@@ -1,29 +1,29 @@
 main.floors.yiqu9=
 {
-"floorId": "yiqu9",
-"title": "主塔 9 层",
-"name": "9",
-"width": 13,
-"height": 13,
-"canFlyTo": true,
-"canFlyFrom": true,
-"canUseQuickShop": true,
-"images": [],
-"ratio": 1,
-"defaultGround": "grass2",
-"firstArrive": [],
-"eachArrive": [],
-"parallelDo": "",
-"events": {},
-"changeFloor": {},
-"beforeBattle": {},
-"afterBattle": {},
-"afterGetItem": {},
-"afterOpenDoor": {},
-"autoEvent": {},
-"cannotMove": {},
-"cannotMoveIn": {},
-"map": [
+    "floorId": "yiqu9",
+    "title": "主塔 9 层",
+    "name": "9",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "X90078",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {},
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
@@ -38,4 +38,16 @@ main.floors.yiqu9=
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
     [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
 ],
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
 }
\ No newline at end of file
diff --git a/project/floors/yushou.js b/project/floors/yushou.js
new file mode 100644
index 0000000..5abf05c
--- /dev/null
+++ b/project/floors/yushou.js
@@ -0,0 +1,62 @@
+main.floors.yushou=
+{
+    "floorId": "yushou",
+    "title": "广场",
+    "name": "1",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "grass2",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "9,0": {
+            "floorId": "guangchang2",
+            "loc": [
+                9,
+                12
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0, 91,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "areas": "牢狱",
+    "bgmap": [
+
+],
+    "fgmap": [
+
+],
+    "bg2map": [
+
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/floors/zhujuejia.js b/project/floors/zhujuejia.js
new file mode 100644
index 0000000..bbee16b
--- /dev/null
+++ b/project/floors/zhujuejia.js
@@ -0,0 +1,106 @@
+main.floors.zhujuejia=
+{
+    "floorId": "zhujuejia",
+    "title": "新建楼层",
+    "name": "0",
+    "width": 13,
+    "height": 13,
+    "canFlyTo": true,
+    "canFlyFrom": true,
+    "canUseQuickShop": true,
+    "cannotViewMap": false,
+    "cannotMoveDirectly": false,
+    "images": [],
+    "ratio": 1,
+    "defaultGround": "ground",
+    "firstArrive": [],
+    "eachArrive": [],
+    "parallelDo": "",
+    "events": {},
+    "changeFloor": {
+        "0,9": {
+            "floorId": "street01",
+            "loc": [
+                12,
+                9
+            ]
+        },
+        "5,2": {
+            "floorId": "Caimhome",
+            "loc": [
+                3,
+                8
+            ]
+        }
+    },
+    "beforeBattle": {},
+    "afterBattle": {},
+    "afterGetItem": {},
+    "afterOpenDoor": {},
+    "autoEvent": {},
+    "cannotMove": {},
+    "cannotMoveIn": {},
+    "map": [
+    [120125,120126,120127,120157,120164,120161,120160,120159,120149,120150,120151,120159,120002],
+    [120133,120134,120135,120165,120172,200550,120168,120167,120157,120158,120159,120167,120010],
+    [120141,120142,120143,120173,120065,200558,120176,120175,120165,120166,120167,140132,140133],
+    [120149,120150,120151,120017,  0,90244,  0,  0,120173,120174,120175,110011,140141],
+    [120157,120158,120159,120017,  0,90244,  0,  0,  0,  0,120021,120002,120002],
+    [120165,120166,120167,110030,  0,  0,  0,120015,120010,120010,120010,120010,120010],
+    [120173,120174,120175,  0,  0,  0,  0,  0,  0,  0,  0,  0,90266],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90266],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,90266],
+    [ 92,  0,  0,  0,  0,  0,200199,  0,  0,  0,  0,140184,140],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,140192,140],
+    [140,140,140186,  0,  0,  0,  0,  0,  0,  0,  0,140192,140],
+    [140,140,140,140,140,140,140,140,140,140,140,140,140]
+],
+    "bgmap": [
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,120175,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078],
+    [90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078,90078]
+],
+    "fgmap": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,90204,  0,140116,140117],
+    [  0,  0,  0,  0,  0, 90,140093,140092,  0,90213,  0,140124,140125],
+    [  0,  0,  0,120038,  0, 90,140099,140100,  0,90223,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,140107,140108,  0,90231,120039,  0,  0],
+    [  0,  0,  0,110022,  0,  0,  0,120007,120007,120007,120007,  0,  0],
+    [140091,140092,140093,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [140099,140100,140101,  0,  0,  0,200175,  0,  0,  0,  0,  0,  0],
+    [140107,140108,140109,  0,  0,  0,200183,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,200191,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,140186,71832,71833,71759,71758,71786,71787,71784,71785,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "bg2map": [
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,120180,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,120073,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,110004,140149],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0],
+    [  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0]
+],
+    "fg2map": [
+
+]
+}
\ No newline at end of file
diff --git a/project/fonts/Verdana.ttf b/project/fonts/Verdana.ttf
new file mode 100644
index 0000000..ea3c090
Binary files /dev/null and b/project/fonts/Verdana.ttf differ
diff --git a/project/functions.js b/project/functions.js
index e90a843..fc50b74 100644
--- a/project/functions.js
+++ b/project/functions.js
@@ -2,55 +2,57 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 {
     "events": {
         "resetGame": function (hero, hard, floorId, maps, values) {
-			// 重置整个游戏;此函数将在游戏开始时,或者每次读档时最先被调用
-			// hero:勇士信息;hard:难度;floorId:当前楼层ID;maps:地图信息;values:全局数值信息
+	// 重置整个游戏;此函数将在游戏开始时,或者每次读档时最先被调用
+	// hero:勇士信息;hard:难度;floorId:当前楼层ID;maps:地图信息;values:全局数值信息
 
-			// 清除游戏数据
-			// 这一步会清空状态栏和全部画布内容,并删除所有动态创建的画布
-			core.clearStatus();
-			// 初始化status
-			core.status = core.clone(core.initStatus, function (name) {
-				return name != 'hero' && name != 'maps';
-			});
-			core.control._bindRoutePush();
-			core.status.played = true;
-			// 初始化人物,图标,统计信息
-			core.status.hero = core.clone(hero);
-			window.hero = core.status.hero;
-			window.flags = core.status.hero.flags;
-			core.events.setHeroIcon(core.status.hero.image, true);
-			core.control._initStatistics(core.animateFrame.totalTime);
-			core.status.hero.statistics.totalTime = core.animateFrame.totalTime =
-				Math.max(core.status.hero.statistics.totalTime, core.animateFrame.totalTime);
-			core.status.hero.statistics.start = null;
-			// 初始难度
-			core.status.hard = hard || "";
-			// 初始化地图
-			core.status.floorId = floorId;
-			core.status.maps = maps;
-			core.maps._resetFloorImages();
-			// 初始化怪物和道具
-			core.material.enemys = core.enemys.getEnemys();
-			core.material.items = core.items.getItems();
-			// 初始化全局数值和全局开关
-			core.values = core.clone(core.data.values);
-			for (var key in values || {})
-				core.values[key] = values[key];
-			core.flags = core.clone(core.data.flags);
-			var globalFlags = core.getFlag("globalFlags", {});
-			for (var key in globalFlags)
-				core.flags[key] = globalFlags[key];
-			core._init_sys_flags();
-			// 初始化界面,状态栏等
-			core.resize();
-			// 状态栏是否显示
-			if (core.hasFlag('hideStatusBar'))
-				core.hideStatusBar(core.hasFlag('showToolbox'));
-			else
-				core.showStatusBar();
-			// 隐藏右下角的音乐按钮
-			core.dom.musicBtn.style.display = 'none';
-		},
+	// 清除游戏数据
+	// 这一步会清空状态栏和全部画布内容,并删除所有动态创建的画布
+	core.clearStatus();
+	core.status.animateObjs = []
+	core.plugin.playing.clear()
+	// 初始化status
+	core.status = core.clone(core.initStatus, function (name) {
+		return name != "hero" && name != "maps";
+	});
+	core.control._bindRoutePush();
+	core.status.played = true;
+	// 初始化人物,图标,统计信息
+	core.status.hero = core.clone(hero);
+	window.hero = core.status.hero;
+	window.flags = core.status.hero.flags;
+	core.events.setHeroIcon(core.status.hero.image, true);
+	core.control._initStatistics(core.animateFrame.totalTime);
+	core.status.hero.statistics.totalTime = core.animateFrame.totalTime =
+		Math.max(
+			core.status.hero.statistics.totalTime,
+			core.animateFrame.totalTime
+		);
+	core.status.hero.statistics.start = null;
+	// 初始难度
+	core.status.hard = hard || "";
+	// 初始化地图
+	core.status.floorId = floorId;
+	core.status.maps = maps;
+	core.maps._resetFloorImages();
+	// 初始化怪物和道具
+	core.material.enemys = core.enemys.getEnemys();
+	core.material.items = core.items.getItems();
+	// 初始化全局数值和全局开关
+	core.values = core.clone(core.data.values);
+	for (var key in values || {}) core.values[key] = values[key];
+	core.flags = core.clone(core.data.flags);
+	var globalFlags = core.getFlag("globalFlags", {});
+	for (var key in globalFlags) core.flags[key] = globalFlags[key];
+	core._init_sys_flags();
+	// 初始化界面,状态栏等
+	core.resize();
+	// 状态栏是否显示
+	if (core.hasFlag("hideStatusBar"))
+		core.hideStatusBar(core.hasFlag("showToolbox"));
+	else core.showStatusBar();
+	// 隐藏右下角的音乐按钮
+	core.dom.musicBtn.style.display = "none";
+},
         "win": function (reason, norank, noexit) {
 	// 游戏获胜事件
 	// 请注意,成绩统计时是按照hp进行上传并排名
@@ -62,117 +64,133 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		core.status.extraEvent = core.clone(core.status.event);
 	}
 
-	// 游戏获胜事件 
+	// 游戏获胜事件
 	core.ui.closePanel();
 	var replaying = core.isReplaying();
 	if (replaying) core.stopReplay();
 	core.waitHeroToStop(function () {
 		if (!noexit) {
-			core.clearMap('all'); // 清空全地图
+			core.status.animateObjs = []
+			core.plugin.playing.clear()
+			core.clearMap("all"); // 清空全地图
 			core.deleteAllCanvas(); // 删除所有创建的画布
 			core.dom.gif2.innerHTML = "";
 		}
 		reason = core.replaceText(reason);
-		core.drawText([
-			"\t[" + (reason || "恭喜通关") + "]你的分数是${status:hp}。"
-		], function () {
-			core.events.gameOver(reason || '', replaying, norank);
-		})
+		core.drawText(
+			["\t[" + (reason || "恭喜通关") + "]你的分数是${status:hp}。"],
+			function () {
+				core.events.gameOver(reason || "", replaying, norank);
+			}
+		);
 	});
 },
         "lose": function (reason) {
-			// 游戏失败事件
-			core.ui.closePanel();
-			var replaying = core.isReplaying();
-			core.stopReplay();
-			core.waitHeroToStop(function () {
-				core.drawText([
-					"\t[" + (reason || "结局1") + "]你死了。\n如题。"
-				], function () {
-					core.events.gameOver(null, replaying);
-				});
-			})
-		},
+	// 游戏失败事件
+	core.ui.closePanel();
+	var replaying = core.isReplaying();
+	core.stopReplay();
+	core.status.animateObjs = []
+	core.plugin.playing.clear()
+	core.waitHeroToStop(function () {
+		core.drawText(
+			["\t[" + (reason || "结局1") + "]你死了。\n如题。"],
+			function () {
+				core.events.gameOver(null, replaying);
+			}
+		);
+	});
+},
         "changingFloor": function (floorId, heroLoc) {
-			// 正在切换楼层过程中执行的操作;此函数的执行时间是“屏幕完全变黑“的那一刻
-			// floorId为要切换到的楼层ID;heroLoc表示勇士切换到的位置
+	// 正在切换楼层过程中执行的操作;此函数的执行时间是“屏幕完全变黑“的那一刻
+	// floorId为要切换到的楼层ID;heroLoc表示勇士切换到的位置
 
-			// ---------- 此时还没有进行切换,当前floorId还是原来的 ---------- //
-			var currentId = core.status.floorId || null; // 获得当前的floorId,可能为null
-			var fromLoad = core.hasFlag('__fromLoad__'); // 是否是读档造成的切换
-			var isFlying = core.hasFlag('__isFlying__'); // 是否是楼传造成的切换
-			if (!fromLoad && !(isFlying && currentId == floorId)) {
-				if (!core.hasFlag("__leaveLoc__")) core.setFlag("__leaveLoc__", {});
-				if (currentId != null) core.getFlag("__leaveLoc__")[currentId] = core.clone(core.status.hero.loc);
+	// ---------- 此时还没有进行切换,当前floorId还是原来的 ---------- //
+	var currentId = core.status.floorId || null; // 获得当前的floorId,可能为null
+	var fromLoad = core.hasFlag("__fromLoad__"); // 是否是读档造成的切换
+	var isFlying = core.hasFlag("__isFlying__"); // 是否是楼传造成的切换
+	if (!fromLoad && !(isFlying && currentId == floorId)) {
+		if (!core.hasFlag("__leaveLoc__")) core.setFlag("__leaveLoc__", {});
+		if (currentId != null)
+			core.getFlag("__leaveLoc__")[currentId] = core.clone(
+				core.status.hero.loc
+			);
+	}
+
+	// 可以对currentId进行判定,比如删除某些自定义图层等
+	// if (currentId == 'MT0') {
+	//     core.deleteAllCanvas();
+	// }
+
+	// 根据分区信息自动砍层与恢复
+	if (core.autoRemoveMaps) core.autoRemoveMaps(floorId);
+
+	// 重置画布尺寸
+	core.maps.resizeMap(floorId);
+	// 设置勇士的位置
+	heroLoc.direction = core.turnDirection(heroLoc.direction);
+	core.status.hero.loc = heroLoc;
+	// 检查重生怪并重置
+	if (!fromLoad) {
+		core.extractBlocks(floorId);
+		core.status.maps[floorId].blocks.forEach(function (block) {
+			if (block.disable && core.enemys.hasSpecial(block.event.id, 23)) {
+				block.disable = false;
+				core.setMapBlockDisabled(floorId, block.x, block.y, false);
+				core.maps._updateMapArray(floorId, block.x, block.y);
 			}
+		});
+		core.control.gatherFollowers();
+	}
 
-			// 可以对currentId进行判定,比如删除某些自定义图层等
-			// if (currentId == 'MT0') {
-			//     core.deleteAllCanvas();
-			// }
+	// ---------- 重绘新地图;这一步将会设置core.status.floorId ---------- //
+	core.drawMap(floorId);
 
-			// 根据分区信息自动砍层与恢复
-			if (core.autoRemoveMaps) core.autoRemoveMaps(floorId);
+	// 切换楼层BGM
+	if (core.status.maps[floorId].bgm) {
+		var bgm = core.status.maps[floorId].bgm;
+		if (bgm instanceof Array)
+			bgm = bgm[Math.floor(Math.random() * bgm.length)]; // 多个bgm则随机播放一个
+		if (!core.hasFlag("__bgm__")) core.playBgm(bgm);
+	} else if (fromLoad && !core.hasFlag("__bgm__")) {
+		core.pauseBgm();
+	}
+	// 更改画面色调
+	var color = core.getFlag("__color__", null);
+	if (!color && core.status.maps[floorId].color)
+		color = core.status.maps[floorId].color;
+	core.clearMap("curtain");
+	core.status.curtainColor = color;
+	if (color)
+		core.fillRect(
+			"curtain",
+			0,
+			0,
+			core._PX_ || core.__PIXELS__,
+			core._PY_ || core.__PIXELS__,
+			core.arrayToRGBA(color)
+		);
+	// 更改天气
+	var weather = core.getFlag("__weather__", null);
+	if (!weather && core.status.maps[floorId].weather)
+		weather = core.status.maps[floorId].weather;
+	if (weather) core.setWeather(weather[0], weather[1]);
+	else core.setWeather();
 
-			// 重置画布尺寸
-			core.maps.resizeMap(floorId);
-			// 设置勇士的位置
-			heroLoc.direction = core.turnDirection(heroLoc.direction);
-			core.status.hero.loc = heroLoc;
-			// 检查重生怪并重置
-			if (!fromLoad) {
-				core.extractBlocks(floorId);
-				core.status.maps[floorId].blocks.forEach(function (block) {
-					if (block.disable && core.enemys.hasSpecial(block.event.id, 23)) {
-						block.disable = false;
-						core.setMapBlockDisabled(floorId, block.x, block.y, false);
-						core.maps._updateMapArray(floorId, block.x, block.y);
-					}
-				});
-				core.control.gatherFollowers();
-			}
-
-			// ---------- 重绘新地图;这一步将会设置core.status.floorId ---------- //
-			core.drawMap(floorId);
-
-			// 切换楼层BGM
-			if (core.status.maps[floorId].bgm) {
-				var bgm = core.status.maps[floorId].bgm;
-				if (bgm instanceof Array) bgm = bgm[Math.floor(Math.random() * bgm.length)]; // 多个bgm则随机播放一个
-				if (!core.hasFlag("__bgm__")) core.playBgm(bgm);
-			} else if (fromLoad && !core.hasFlag("__bgm__")) {
-				core.pauseBgm();
-			}
-			// 更改画面色调
-			var color = core.getFlag('__color__', null);
-			if (!color && core.status.maps[floorId].color)
-				color = core.status.maps[floorId].color;
-			core.clearMap('curtain');
-			core.status.curtainColor = color;
-			if (color) core.fillRect('curtain', 0, 0, core._PX_ || core.__PIXELS__, core._PY_ || core.__PIXELS__, core.arrayToRGBA(color));
-			// 更改天气
-			var weather = core.getFlag('__weather__', null);
-			if (!weather && core.status.maps[floorId].weather)
-				weather = core.status.maps[floorId].weather;
-			if (weather)
-				core.setWeather(weather[0], weather[1]);
-			else core.setWeather();
-
-			// ...可以新增一些其他内容,比如创建个画布在右上角显示什么内容等等
-
-		},
+	// ...可以新增一些其他内容,比如创建个画布在右上角显示什么内容等等
+},
         "afterChangeFloor": function (floorId) {
 	// 转换楼层结束的事件;此函数会在整个楼层切换完全结束后再执行
 	// floorId是切换到的楼层
 
 	// 如果是读档,则进行检查(是否需要恢复事件)
-	if (core.hasFlag('__fromLoad__')) {
+	if (core.hasFlag("__fromLoad__")) {
 		core.events.recoverEvents(core.getFlag("__events__"));
 		core.removeFlag("__events__");
 	} else {
 		// 每次抵达楼层执行的事件
 		core.insertAction(core.floors[floorId].eachArrive);
-		core.ui.statusBar._update_map()
+		core.ui.statusBar._update_map();
 		// 首次抵达楼层时执行的事件(后插入,先执行)
 		if (!core.hasVisitedFloor(floorId)) {
 			core.insertAction(core.floors[floorId].firstArrive);
@@ -180,6 +198,7 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 			core.plugin.bfs();
 		}
 	}
+	if (!core.isReplaying()) core.plugin.drawCommentSign()
 },
         "flyTo": function (toId, callback) {
 	// 楼层传送器的使用,从当前楼层飞往toId
@@ -188,9 +207,13 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	var fromId = core.status.floorId;
 
 	// 检查能否飞行
-	if (!core.status.maps[fromId].canFlyFrom || !core.status.maps[toId].canFlyTo || !core.hasVisitedFloor(toId)) {
-		core.playSound('操作失败');
-		core.drawTip("无法飞往" + core.status.maps[toId].title + "!", 'fly');
+	if (
+		!core.status.maps[fromId].canFlyFrom ||
+		!core.status.maps[toId].canFlyTo ||
+		!core.hasVisitedFloor(toId)
+	) {
+		core.playSound("操作失败");
+		core.drawTip("无法飞往" + core.status.maps[toId].title + "!", "fly");
 		return false;
 	}
 
@@ -200,8 +223,11 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	if (core.flags.flyRecordPosition) {
 		loc = core.getFlag("__leaveLoc__", {})[toId] || null;
 	}
-	if (core.status.maps[toId].flyPoint != null && core.status.maps[toId].flyPoint.length == 2) {
-		stair = 'flyPoint';
+	if (
+		core.status.maps[toId].flyPoint != null &&
+		core.status.maps[toId].flyPoint.length == 2
+	) {
+		stair = "flyPoint";
 	}
 	if (stair == null && loc == null) {
 		// 获得两个楼层的索引,以决定是上楼梯还是下楼梯
@@ -209,14 +235,15 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 			toIndex = core.floorIds.indexOf(toId);
 		var stair = fromIndex <= toIndex ? "downFloor" : "upFloor";
 		// 地下层:同层传送至上楼梯
-		if (fromIndex == toIndex && core.status.maps[fromId].underGround) stair = "upFloor";
+		if (fromIndex == toIndex && core.status.maps[fromId].underGround)
+			stair = "upFloor";
 	}
 
 	// 记录录像
 	core.status.route.push("fly:" + toId);
 	// 传送
 	core.ui.closePanel();
-	core.setFlag('__isFlying__', true);
+	core.setFlag("__isFlying__", true);
 	core.changeFloor(toId, stair, loc, null, function () {
 		core.removeFlag("__isFlying__");
 		if (callback) callback();
@@ -225,42 +252,47 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	return true;
 },
         "beforeBattle": function (enemyId, x, y) {
-			// 战斗前触发的事件,可以加上一些战前特效(详见下面支援的例子)
-			// 此函数在“检测能否战斗和自动存档”【之后】执行。如果需要更早的战前事件,请在插件中覆重写 core.events.doSystemEvent 函数。
-			// 返回true则将继续战斗,返回false将不再战斗。
+      // 战斗前触发的事件,可以加上一些战前特效(详见下面支援的例子)
+      // 此函数在“检测能否战斗和自动存档”【之后】执行。如果需要更早的战前事件,请在插件中覆重写 core.events.doSystemEvent 函数。
+      // 返回true则将继续战斗,返回false将不再战斗。
 
-			// ------ 支援技能 ------ //
-			if (x != null && y != null) {
-				var index = x + "," + y,
-					cache = core.status.checkBlock.cache[index] || {},
-					guards = cache.guards || [];
-				// 如果存在支援怪
-				if (guards.length > 0) {
-					// 记录flag,当前要参与支援的怪物
-					core.setFlag("__guards__" + x + "_" + y, guards);
-					var actions = [{ "type": "playSound", "name": "跳跃" }];
-					// 增加支援的特效动画(图块跳跃)
-					guards.forEach(function (g) {
-						core.push(actions, { "type": "jump", "from": [g[0], g[1]], "to": [x, y], "time": 300, "keep": false, "async": true });
-					});
-					core.push(actions, [
-						{ "type": "waitAsync" }, // 等待所有异步事件执行完毕
-						{ "type": "setBlock", "number": enemyId, "loc": [[x, y]] }, // 重新设置怪物自身
-						{ "type": "battle", "loc": [x, y] } // 重要!重新触发本次战斗
-					]);
-					core.insertAction(actions);
-					return false;
-				}
-			}
+      // ------ 支援技能 ------ //
+      if (x != null && y != null) {
+        var index = x + "," + y,
+          cache = core.status.checkBlock.cache[index] || {},
+          guards = cache.guards || [];
+        // 如果存在支援怪
+        if (guards.length > 0) {
+          // 记录flag,当前要参与支援的怪物
+          core.setFlag("__guards__" + x + "_" + y, guards);
+          var actions = [{ type: "playSound", name: "跳跃" }];
+          // 增加支援的特效动画(图块跳跃)
+          guards.forEach(function (g) {
+            core.push(actions, {
+              type: "jump",
+              from: [g[0], g[1]],
+              to: [x, y],
+              time: 300,
+              keep: false,
+              async: true,
+            });
+          });
+          core.push(actions, [
+            { type: "waitAsync" }, // 等待所有异步事件执行完毕
+            { type: "setBlock", number: enemyId, loc: [[x, y]] }, // 重新设置怪物自身
+            { type: "battle", loc: [x, y] }, // 重要!重新触发本次战斗
+          ]);
+          core.insertAction(actions);
+          return false;
+        }
+      }
 
-			return true;
-		},
-        "afterBattle": function (enemyId, x, y) {
+      return true;
+    },
+        "afterBattle": async function (enemyId, x, y) {
 	// 战斗结束后触发的事件
-
-	var enemy = core.material.enemys[enemyId];
-	var special = enemy.special;
-
+	var enemy = core.material.enemys[enemyId]
+	var special = core.getEnemyValue(enemy, "special", x, y);
 	// 播放战斗音效和动画
 	// 默认播放的动画;你也可以使用
 	var animate = 'hand'; // 默认动画
@@ -270,19 +302,30 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		animate = core.material.items[equipId].equip.animate;
 	// 你也可以在这里根据自己的需要,比如enemyId或special或flag来修改播放的动画效果
 	// if (enemyId == '...') animate = '...';
+	if (core.getFlag('noAnimate')) {
+		// 检查该动画是否存在SE,如果不存在则使用默认音效
+		if (!(core.material.animates[animate] || {}).se)
+			core.playSound('attack.mp3');
 
-	// 检查该动画是否存在SE,如果不存在则使用默认音效
-	if (!(core.material.animates[animate] || {}).se)
-		core.playSound('attack.mp3');
-
-	// 播放动画;如果不存在坐标(强制战斗)则播放到勇士自身
-	if (x != null && y != null)
-		core.drawAnimate(animate, x, y);
-	else
-		core.drawHeroAnimate(animate);
-
+		// 播放动画;如果不存在坐标(强制战斗)则播放到勇士自身
+		if (x != null && y != null)
+			core.drawAnimate(animate, x, y);
+		else
+			core.drawHeroAnimate(animate);
+	}
 	// 获得战斗伤害信息
 	var damageInfo = core.getDamageInfo(enemyId, null, x, y) || {};
+	if ((!core.getFlag("noAnimate") || damageInfo.damage >= core.status.hero.hp) && (!main.replayChecking && !core.isReplaying())) await core.attackAnimate(
+		enemyId,
+		damageInfo.start[0],
+		damageInfo.start[1],
+		damageInfo.start[2],
+		damageInfo.start[3],
+		damageInfo.heroDiffList,
+		damageInfo.enemyDiffList,
+		damageInfo.heroanimateList,
+		damageInfo.enemyanimateList
+	)
 	// 战斗伤害
 	var damage = damageInfo.damage;
 	// 当前战斗回合数,可用于战后所需的判定
@@ -354,8 +397,8 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	}
 	// 退化
 	if (core.enemys.hasSpecial(special, 21)) {
-		core.status.hero.atk -= (enemy.atkValue || 0);
-		core.status.hero.def -= (enemy.defValue || 0);
+		core.status.hero.atk -= core.getEnemyValue(enemy, "atkValue", x, y) || 0;
+		core.status.hero.def -= core.getEnemyValue(enemy, "defValue", x, y) || 0;
 		if (core.status.hero.atk < 0) core.status.hero.atk = 0;
 		if (core.status.hero.def < 0) core.status.hero.def = 0;
 	}
@@ -410,7 +453,7 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	// 因为removeBlock和hideBlock都会刷新状态栏,因此将删除部分移动到这里并保证刷新只执行一次,以提升效率
 	if (core.getBlock(x, y) != null) {
 		// 检查是否是重生怪物;如果是则仅隐藏不删除
-		if (core.hasSpecial(enemy.special, 23)) {
+		if (core.hasSpecial(special, 23)) {
 			core.hideBlock(x, y);
 		} else {
 			core.removeBlock(x, y);
@@ -427,61 +470,72 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 
 },
         "afterOpenDoor": function (doorId, x, y) {
-			// 开一个门后触发的事件
+      // 开一个门后触发的事件
 
-			var todo = [];
-			// 检查该点的开门后事件
-			if (core.status.floorId) {
-				core.push(todo, core.floors[core.status.floorId].afterOpenDoor[x + "," + y]);
-			}
-			// 检查批量开门事件
-			var door = core.getBlockById(doorId);
-			if (door && door.event.doorInfo) {
-				core.push(todo, door.event.doorInfo.afterOpenDoor);
-			}
+      var todo = [];
+      // 检查该点的开门后事件
+      if (core.status.floorId) {
+        core.push(
+          todo,
+          core.floors[core.status.floorId].afterOpenDoor[x + "," + y]
+        );
+      }
+      // 检查批量开门事件
+      var door = core.getBlockById(doorId);
+      if (door && door.event.doorInfo) {
+        core.push(todo, door.event.doorInfo.afterOpenDoor);
+      }
 
-			if (todo.length > 0) core.insertAction(todo, x, y);
+      if (todo.length > 0) core.insertAction(todo, x, y);
 
-			if (core.status.event.id == null)
-				core.continueAutomaticRoute();
-			else
-				core.clearContinueAutomaticRoute();
-		},
+      if (core.status.event.id == null) core.continueAutomaticRoute();
+      else core.clearContinueAutomaticRoute();
+    },
         "afterGetItem": function (itemId, x, y, isGentleClick) {
-			// 获得一个道具后触发的事件
-			// itemId:获得的道具ID;x和y是该道具所在的坐标
-			// isGentleClick:是否是轻按触发的
-			if (itemId.endsWith('Potion') && core.material.items[itemId].cls == 'items')
-				core.playSound('回血');
-			else if (itemId.endsWith('Gem') && core.material.items[itemId].cls == 'items')
-				core.playSound('宝石')
-			else
-				core.playSound('获得道具');
+      // 获得一个道具后触发的事件
+      // itemId:获得的道具ID;x和y是该道具所在的坐标
+      // isGentleClick:是否是轻按触发的
+      if (
+        itemId.endsWith("Potion") &&
+        core.material.items[itemId].cls == "items"
+      )
+        core.playSound("回血");
+      else if (
+        itemId.endsWith("Gem") &&
+        core.material.items[itemId].cls == "items"
+      )
+        core.playSound("宝石");
+      else core.playSound("获得道具");
 
-			var todo = [];
-			// 检查该点的获得道具后事件。
-			if (core.status.floorId == null) return;
-			var event = core.floors[core.status.floorId].afterGetItem[x + "," + y];
-			if (event && (event instanceof Array || !isGentleClick || !event.disableOnGentleClick)) {
-				if (event.data) event = event.data;
-				core.unshift(todo, event);
-			}
+      var todo = [];
+      // 检查该点的获得道具后事件。
+      if (core.status.floorId == null) return;
+      var event = core.floors[core.status.floorId].afterGetItem[x + "," + y];
+      if (
+        event &&
+        (event instanceof Array ||
+          !isGentleClick ||
+          !event.disableOnGentleClick)
+      ) {
+        if (event.data) event = event.data;
+        core.unshift(todo, event);
+      }
 
-			if (todo.length > 0) core.insertAction(todo, x, y);
-		},
+      if (todo.length > 0) core.insertAction(todo, x, y);
+    },
         "afterPushBox": function () {
-			// 推箱子后的事件
-			if (core.searchBlock('box').length == 0) {
-				// 可以通过if语句来进行开门操作
-				/*
+      // 推箱子后的事件
+      if (core.searchBlock("box").length == 0) {
+        // 可以通过if语句来进行开门操作
+        /*
 				if (core.status.floorId=='xxx') { // 在某个楼层
 					core.insertAction([ // 插入一条事件
 						{"type": "openDoor", "loc": [x,y]} // 开门
 					])
 				}
 				*/
-			}
-		}
+      }
+    }
     },
     "enemys": {
         "getSpecials": function () {
@@ -491,36 +545,221 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	// 第五项为该特殊属性的标记;目前 1 代表是地图类技能(需要进行遍历全图)
 	// 名字和描述可以直接写字符串,也可以写个function将怪物传进去
 	return [
-		[1, "先攻", "怪物首先攻击", "#ffcc33"],
-		[3, "坚固", "怪物防御不小于角色攻击-1", "#c0b088"],
-		[6, function (enemy) { return (enemy.n || '') + "连击"; }, function (enemy) { return "怪物每回合攻击" + (enemy.n || 4) + "次"; }, "#ffee77"],
-		[7, "破甲", function (enemy) { return "战斗前,怪物附加角色防御的" + Math.floor(100 * (enemy.breakArmor || core.values.breakArmor || 0)) + "%作为伤害"; }, "#88c0ff"],
-		[8, "反击", function (enemy) { return "战斗时,怪物每回合附加角色攻击的" + Math.floor(100 * (enemy.counterAttack || core.values.counterAttack || 0)) + "%作为伤害,无视角色防御"; }, "#ffaa44"],
-		[9, "净化", function (enemy) { return "战斗前,怪物附加角色护盾的" + (enemy.purify || core.values.purify) + "倍作为伤害"; }, "#80eed6"],
+		[1, "先攻", "怪物首先攻击", "#ffcc33"], //√
+		//[2, "魔攻", "怪物无视角色的防御", "#bbb0ff"],
+		[3, "坚固", "怪物防御不小于角色攻击-1", "#c0b088"], //√
+		[
+			6,
+			function (enemy) {
+				return (enemy.n || "") + "连击";
+			},
+			function (enemy) {
+				return "怪物每回合攻击" + (enemy.n || 4) + "次";
+			},
+			"#ffee77",
+		], //想改成“以XX%、XX%攻击力各攻击一次”
+		[
+			7,
+			"破甲",
+			function (enemy) {
+				return (
+					"怪物无视角色防御的" +
+					Math.floor(
+						100 * (enemy.breakArmor || core.values.breakArmor || 0)
+					) +
+					"%"
+				);
+			},
+			"#88c0ff",
+		], //√
+		[
+			8,
+			"反击",
+			function (enemy) {
+				return (
+					"战斗时,怪物每回合附加角色攻击的" +
+					Math.floor(
+						100 * (enemy.counterAttack || core.values.counterAttack || 0)
+					) +
+					"%作为伤害,无视角色防御"
+				);
+			},
+			"#ffaa44",
+		],
+		[
+			9,
+			"净化",
+			function (enemy) {
+				return (
+					"战斗前,怪物附加角色护盾的" +
+					(enemy.purify || core.values.purify) +
+					"倍作为伤害"
+				);
+			},
+			"#80eed6",
+		],
 		[10, "模仿", "怪物的攻防和角色攻防相等", "#b0c0dd"],
-		[11, "吸血", function (enemy) { return "战斗前,怪物首先吸取角色的" + Math.floor(100 * enemy.vampire || 0) + "%生命(约" + Math.floor((enemy.vampire || 0) * core.getStatus('hp')) + "点)作为伤害" + (enemy.add ? ",并把伤害数值加到自身生命上" : ""); }, "#dd4448"],
-		[12, "中毒", "战斗后,角色陷入中毒状态,每一步损失生命" + core.values.poisonDamage + "点", "#99ee88"],
-		[13, "衰弱", "战斗后,角色陷入衰弱状态,攻防暂时下降" + (core.values.weakValue >= 1 ? core.values.weakValue + "点" : parseInt(core.values.weakValue * 100) + "%"), "#f0bbcc"],
-		[14, "诅咒", "战斗后,角色陷入诅咒状态,战斗无法获得金币和经验", "#bbeef0"],
-		[15, "领域", function (enemy) { return "经过怪物周围" + (enemy.zoneSquare ? "九宫格" : "十字") + "范围内" + (enemy.range || 1) + "格时自动减生命" + (enemy.zone || 0) + "点"; }, "#c677dd"],
+		[
+			11,
+			"吸血",
+			function (enemy) {
+				return (
+					"战斗前,怪物首先吸取角色的" +
+					Math.floor(100 * enemy.vampire || 0) +
+					"%生命(约" +
+					Math.floor((enemy.vampire || 0) * core.getStatus("hp")) +
+					"点)作为伤害" +
+					(enemy.add ? ",并把伤害数值加到自身生命上" : "")
+				);
+			},
+			"#dd4448",
+		],
+		[
+			12,
+			"中毒",
+			"战斗后,角色陷入中毒状态,每一步损失生命" +
+			core.values.poisonDamage +
+			"点",
+			"#99ee88",
+		],
+		[
+			13,
+			"衰弱",
+			"战斗后,角色陷入衰弱状态,攻防暂时下降" +
+			(core.values.weakValue >= 1 ?
+				core.values.weakValue + "点" :
+				parseInt(core.values.weakValue * 100) + "%"),
+			"#f0bbcc",
+		],
+		[
+			14,
+			"诅咒",
+			"战斗后,角色陷入诅咒状态,战斗无法获得金币和经验",
+			"#bbeef0",
+		],
+		[
+			15,
+			"领域",
+			function (enemy) {
+				return (
+					"经过怪物周围" +
+					(enemy.zoneSquare ? "九宫格" : "十字") +
+					"范围内" +
+					(enemy.range || 1) +
+					"格时自动减生命" +
+					(enemy.zone || 0) +
+					"点"
+				);
+			},
+			"#c677dd",
+		],
 		[16, "夹击", "经过两只相同的怪物中间,角色生命值变成一半", "#bb99ee"],
-		[17, "仇恨", "战斗前,怪物附加之前积累的仇恨值作为伤害;战斗后,释放一半的仇恨值。(每杀死一个怪物获得" + (core.values.hatred || 0) + "点仇恨值)", "#b0b666"],
-		[18, "阻击", function (enemy) { return "经过怪物周围" + (enemy.zoneSquare ? "九宫格" : "十字") + "时自动减生命" + (enemy.repulse || 0) + "点,同时怪物后退一格"; }, "#8888e6"],
+		[
+			17,
+			"仇恨",
+			"战斗前,怪物附加之前积累的仇恨值作为伤害;战斗后,释放一半的仇恨值。(每杀死一个怪物获得" +
+			(core.values.hatred || 0) +
+			"点仇恨值)",
+			"#b0b666",
+		],
+		[
+			18,
+			"阻击",
+			function (enemy) {
+				return (
+					"经过怪物周围" +
+					(enemy.zoneSquare ? "九宫格" : "十字") +
+					"时自动减生命" +
+					(enemy.repulse || 0) +
+					"点,同时怪物后退一格"
+				);
+			},
+			"#8888e6",
+		],
 		[19, "自爆", "战斗后角色的生命值变成1", "#ff6666"],
 		[20, "无敌", "角色无法打败怪物,除非拥有十字架", "#aaaaaa"],
-		[21, "退化", function (enemy) { return "战斗后角色永久下降" + (enemy.atkValue || 0) + "点攻击和" + (enemy.defValue || 0) + "点防御"; }],
-		[22, "固伤", function (enemy) { return "战斗前,怪物对角色造成" + (enemy.damage || 0) + "点固定伤害,未开启负伤时无视角色护盾。"; }, "#ff9977"],
+		[
+			21,
+			"退化",
+			function (enemy) {
+				return (
+					"战斗后角色永久下降" +
+					(enemy.atkValue || 0) +
+					"点攻击和" +
+					(enemy.defValue || 0) +
+					"点防御"
+				);
+			},
+		],
+		[
+			22,
+			"固伤",
+			function (enemy) {
+				return (
+					"战斗前,怪物对角色造成" +
+					(enemy.damage || 0) +
+					"点固定伤害,未开启负伤时无视角色护盾。"
+				);
+			},
+			"#ff9977",
+		],
 		[23, "重生", "怪物被击败后,角色转换楼层则怪物将再次出现", "#a0e0ff"],
-		[24, "激光", function (enemy) { return "经过怪物同行或同列时自动减生命" + (enemy.laser || 0) + "点"; }, "#dda0dd"],
-		[25, "光环", function (enemy) { return (enemy.range != null ? ((enemy.haloSquare ? "该怪物九宫格" : "该怪物十字") + enemy.haloRange + "格范围内") : "同楼层所有") + "怪物生命提升" + (enemy.hpBuff || 0) + "%,攻击提升" + (enemy.atkBuff || 0) + "%,防御提升" + (enemy.defBuff || 0) + "%," + (enemy.haloAdd ? "可叠加" : "不可叠加"); }, "#e6e099", 1],
-		[26, "支援", "当周围一圈的怪物受到攻击时将上前支援,并组成小队战斗。", "#77c0b6", 1],
-		[27, "捕捉", function (enemy) { return "当走到怪物周围" + (enemy.zoneSquare ? "九宫格" : "十字") + "时会强制进行战斗。"; }, "#c0ddbb"]
+		[
+			24,
+			"激光",
+			function (enemy) {
+				return "经过怪物同行或同列时自动减生命" + (enemy.laser || 0) + "点";
+			},
+			"#dda0dd",
+		],
+		[
+			25,
+			"光环",
+			function (enemy) {
+				return (
+					(enemy.range != null ?
+						(enemy.haloSquare ? "该怪物九宫格" : "该怪物十字") +
+						enemy.haloRange +
+						"格范围内" :
+						"同楼层所有") +
+					"怪物生命提升" +
+					(enemy.hpBuff || 0) +
+					"%,攻击提升" +
+					(enemy.atkBuff || 0) +
+					"%,防御提升" +
+					(enemy.defBuff || 0) +
+					"%," +
+					(enemy.haloAdd ? "可叠加" : "不可叠加")
+				);
+			},
+			"#e6e099",
+			1,
+		],
+		[
+			26,
+			"支援",
+			"当周围一圈的怪物受到攻击时将上前支援,并组成小队战斗。",
+			"#77c0b6",
+			1,
+		],
+		[
+			27,
+			"捕捉",
+			function (enemy) {
+				return (
+					"当走到怪物周围" +
+					(enemy.zoneSquare ? "九宫格" : "十字") +
+					"时会强制进行战斗。"
+				);
+			},
+			"#c0ddbb",
+		],
 	];
 },
         "getEnemyInfo": function (enemy, hero, x, y, floorId) {
 	// 获得某个怪物变化后的数据;该函数将被伤害计算和怪物手册使用
 	// 例如:坚固、模仿、仿攻等等
-	// 
+	//
 	// 参数说明:
 	// enemy:该怪物信息
 	// hero_hp,hero_atk,hero_def,hero_mdef:勇士的生命攻防护盾数据
@@ -528,18 +767,25 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	// floorId:该怪物所在的楼层
 	// 后面三个参数主要是可以在光环等效果上可以适用(也可以按需制作部分范围光环效果)
 	floorId = floorId || core.status.floorId;
-	var hero_hp = core.getRealStatusOrDefault(hero, 'hp'),
-		hero_atk = core.getRealStatusOrDefault(hero, 'atk'),
-		hero_def = core.getRealStatusOrDefault(hero, 'def'),
-		hero_mdef = core.getRealStatusOrDefault(hero, 'mdef');
-
-	var mon_hp = core.getEnemyValue(enemy, 'hp', x, y, floorId),
-		mon_atk = core.getEnemyValue(enemy, 'atk', x, y, floorId),
-		mon_def = core.getEnemyValue(enemy, 'def', x, y, floorId),
-		mon_special = core.getEnemyValue(enemy, 'special', x, y, floorId);
-	var mon_money = core.getEnemyValue(enemy, 'money', x, y, floorId),
-		mon_exp = core.getEnemyValue(enemy, 'exp', x, y, floorId),
-		mon_point = core.getEnemyValue(enemy, 'point', x, y, floorId);
+	var hero_hp = core.getRealStatusOrDefault(hero, "hp"),
+		hero_atk = core.getRealStatusOrDefault(hero, "atk"),
+		hero_def = core.getRealStatusOrDefault(hero, "def"),
+		hero_mdef = core.getRealStatusOrDefault(hero, "mdef"),
+		hero_speed = core.getRealStatusOrDefault(hero, "speed");
+	var mon_id = core.getEnemyValue(enemy, "id", x, y, floorId),
+		mon_name = core.getEnemyValue(enemy, "name", x, y, floorId);
+	var mon_hp = core.getEnemyValue(enemy, "hp", x, y, floorId),
+		mon_atk = core.getEnemyValue(enemy, "atk", x, y, floorId),
+		mon_def = core.getEnemyValue(enemy, "def", x, y, floorId),
+		mon_mdef = core.getEnemyValue(enemy, "mdef", x, y, floorId) || 0,
+		mon_spell = core.getEnemyValue(enemy, "spell", x, y, floorId) || 0,
+		mon_speed = core.getEnemyValue(enemy, "speed", x, y, floorId) || 1,
+		mon_special = core.getEnemyValue(enemy, "special", x, y, floorId);
+	var mon_money = core.getEnemyValue(enemy, "money", x, y, floorId),
+		mon_exp = core.getEnemyValue(enemy, "exp", x, y, floorId),
+		mon_point = core.getEnemyValue(enemy, "point", x, y, floorId);
+	var mon_barrier = 0,
+		mon_absorb_damage = 0;
 	// 模仿
 	if (core.hasSpecial(mon_special, 10)) {
 		mon_atk = hero_atk;
@@ -551,7 +797,6 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	}
 
 	var guards = [];
-
 	// 光环和支援检查
 	if (!core.status.checkBlock) core.status.checkBlock = {};
 
@@ -563,7 +808,7 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		// 已经计算过的光环怪ID列表,用于判定叠加
 		var usedEnemyIds = {};
 		// 检查光环和支援的缓存
-		var index = x != null && y != null ? (x + "," + y) : floorId;
+		var index = x != null && y != null ? x + "," + y : floorId;
 		if (!core.status.checkBlock.cache) core.status.checkBlock.cache = {};
 		var cache = core.status.checkBlock.cache[index];
 		if (!cache) {
@@ -583,7 +828,12 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 								dy = Math.abs(block.y - y);
 							// 检查十字和九宫格光环
 							if (dx + dy <= enemy.haloRange) inRange = true;
-							if (enemy.haloSquare && dx <= enemy.haloRange && dy <= enemy.haloRange) inRange = true;
+							if (
+								enemy.haloSquare &&
+								dx <= enemy.haloRange &&
+								dy <= enemy.haloRange
+							)
+								inRange = true;
 						}
 						// 检查是否可叠加
 						if (inRange && (enemy.haloAdd || !usedEnemyIds[enemy.id])) {
@@ -594,10 +844,17 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 						}
 					}
 					// 检查【支援】技能,数字26
-					if (enemy && core.hasSpecial(enemy.special, 26) &&
+					if (
+						enemy &&
+						core.hasSpecial(enemy.special, 26) &&
 						// 检查支援条件,坐标存在,距离为1,且不能是自己
 						// 其他类型的支援怪,比如十字之类的话.... 看着做是一样的
-						x != null && y != null && Math.abs(block.x - x) <= 1 && Math.abs(block.y - y) <= 1 && !(x == block.x && y == block.y)) {
+						x != null &&
+						y != null &&
+						Math.abs(block.x - x) <= 1 &&
+						Math.abs(block.y - y) <= 1 &&
+						!(x == block.x && y == block.y)
+					) {
 						// 记录怪物的x,y,ID
 						guards.push([block.x, block.y, id]);
 					}
@@ -607,7 +864,12 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 				}
 			});
 
-			core.status.checkBlock.cache[index] = { "hp_buff": hp_buff, "atk_buff": atk_buff, "def_buff": def_buff, "guards": guards };
+			core.status.checkBlock.cache[index] = {
+				hp_buff: hp_buff,
+				atk_buff: atk_buff,
+				def_buff: def_buff,
+				guards: guards,
+			};
 		} else {
 			// 直接使用缓存数据
 			hp_buff = cache.hp_buff;
@@ -617,9 +879,9 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		}
 
 		// 增加比例;如果要增加数值可以直接在这里修改
-		mon_hp *= (1 + hp_buff / 100);
-		mon_atk *= (1 + atk_buff / 100);
-		mon_def *= (1 + def_buff / 100);
+		mon_hp *= 1 + hp_buff / 100;
+		mon_atk *= 1 + atk_buff / 100;
+		mon_def *= 1 + def_buff / 100;
 	}
 
 	// TODO:可以在这里新增其他的怪物数据变化
@@ -630,19 +892,27 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	// 也可以按需增加各种自定义内容
 
 	return {
-		"hp": Math.floor(mon_hp),
-		"atk": Math.floor(mon_atk),
-		"def": Math.floor(mon_def),
-		"money": Math.floor(mon_money),
-		"exp": Math.floor(mon_exp),
-		"point": Math.floor(mon_point),
-		"special": mon_special,
-		"guards": guards, // 返回支援情况
+		id: mon_id,
+		name: mon_name,
+		hp: Math.floor(mon_hp),
+		atk: Math.floor(mon_atk),
+		def: Math.floor(mon_def),
+		mdef: Math.floor(mon_mdef),
+		spell: Math.floor(mon_spell),
+		speed: Math.floor(mon_speed),
+		barrier: Math.floor(mon_barrier),
+		absorb: Math.floor(mon_absorb_damage),
+		money: Math.floor(mon_money),
+		exp: Math.floor(mon_exp),
+		point: Math.floor(mon_point),
+		special: mon_special,
+		guards: guards // 返回支援情况
+
 	};
 },
         "getDamageInfo": function (enemy, hero, x, y, floorId) {
 	// 获得战斗伤害信息(实际伤害计算函数)
-	// 
+	//
 	// 参数说明:
 	// enemy:该怪物信息
 	// hero:勇士的当前数据;如果对应项不存在则会从core.status.hero中取。
@@ -651,43 +921,76 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	// 后面三个参数主要是可以在光环等效果上可以适用
 	floorId = floorId || core.status.floorId;
 
-	var hero_hp = core.getRealStatusOrDefault(hero, 'hp'),
-		hero_atk = core.getRealStatusOrDefault(hero, 'atk'),
-		hero_def = core.getRealStatusOrDefault(hero, 'def'),
-		hero_mdef = core.getRealStatusOrDefault(hero, 'mdef'),
-		origin_hero_hp = core.getStatusOrDefault(hero, 'hp'),
-		origin_hero_atk = core.getStatusOrDefault(hero, 'atk'),
-		origin_hero_def = core.getStatusOrDefault(hero, 'def');
-
-	// 勇士的负属性都按0计算
-	hero_hp = Math.max(0, hero_hp);
-	hero_atk = Math.max(0, hero_atk);
-	hero_def = Math.max(0, hero_def);
-	hero_mdef = Math.max(0, hero_mdef);
+	var hero_hp = core.getRealStatusOrDefault(hero, "hp"),
+		hero_atk = core.getRealStatusOrDefault(hero, "atk"),
+		hero_def = core.getRealStatusOrDefault(hero, "def"),
+		hero_matk = core.getRealStatusOrDefault(hero, "matk"),
+		hero_mdef = core.getRealStatusOrDefault(hero, "mdef"),
+		hero_mhp = core.getRealStatusOrDefault(hero, "mhp"),
+		hero_speed = core.getRealStatusOrDefault(hero, "speed"),
+		hero_spell = core.getRealStatusOrDefault(hero, "spell"),
+		origin_hero_hp = core.getStatusOrDefault(hero, "hp"),
+		origin_hero_atk = core.getStatusOrDefault(hero, "atk"),
+		origin_hero_def = core.getStatusOrDefault(hero, "def");
 
+	//编辑器特判
+	if (main.mode == "editor") {
+		hero_hp = hero?.hp ?? core.status.hero.hp,
+			hero_atk = hero?.atk ?? core.status.hero.atk,
+			hero_def = hero?.def ?? core.status.hero.def,
+			hero_matk = hero?.matk ?? core.status.hero.matk,
+			hero_mdef = hero?.mdef ?? core.status.hero.mdef,
+			hero_mhp = hero?.mhp ?? core.status.hero.mhp,
+			hero_speed = hero?.speed ?? core.status.hero.speed,
+			hero_spell = hero?.spell ?? core.status.hero.spell;
+	}
 	// 怪物的各项数据
 	// 对坚固模仿等处理扔到了脚本编辑-getEnemyInfo之中
 	var enemyInfo = core.enemys.getEnemyInfo(enemy, hero, x, y, floorId);
 	var mon_hp = enemyInfo.hp,
 		mon_atk = enemyInfo.atk,
 		mon_def = enemyInfo.def,
-		mon_special = enemyInfo.special;
-
+		mon_mdef = enemyInfo.mdef,
+		mon_spell = enemyInfo.spell,
+		mon_speed = enemyInfo.speed,
+		mon_special = enemyInfo.special,
+		mon_absorb_damage = enemyInfo.absorb,
+		mon_barrier = enemyInfo.barrier;
+	const { lcm, gcd } = core.plugin.utils
+	const equip0 = core.getEquip(0)
+	//---第一部分:静态属性修正---
+	//此处写入静态影响勇士属性的勇士或怪物技能(静态影响怪物属性的技能于getEnemyInfo中写入)
 	// 技能的处理
-	if (core.getFlag('skill', 0) == 1) { // 开启了技能1:二倍斩
-		hero_atk *= 2; // 计算时攻击力翻倍	
+	if (core.getFlag("skill", 0) == 1) {
+		// 开启了技能1:二倍斩
+		hero_atk *= 2; // 计算时攻击力翻倍
 	}
 
+	// 破甲
+	if (core.hasSpecial(mon_special, 7))
+		hero_def -= Math.floor(
+			(enemy.breakArmor || core.values.breakArmor) * hero_def
+		);
+
+	//勇士属性取整
+	hero_atk = Math.max(0, Math.floor(hero_atk));
+	hero_def = Math.max(0, Math.floor(hero_def));
+	hero_speed = Math.max(0, Math.floor(hero_speed));
+	hero_spell = Math.max(0, Math.floor(hero_spell));
+	hero_matk = Math.min(100, Math.max(0, Math.floor(hero_matk)));
+	hero_mdef = Math.min(100, Math.max(0, Math.floor(hero_mdef)));
+	hero_mhp = Math.min(100, Math.max(0, Math.floor(hero_mhp)));
+
 	// 如果是无敌属性,且勇士未持有十字架
 	if (core.hasSpecial(mon_special, 20) && !core.hasItem("cross"))
 		return null; // 不可战斗
 
 	// 战前造成的额外伤害(可被护盾抵消)
-	var init_damage = 0;
+	let init_damage = 0;
 
 	// 吸血
 	if (core.hasSpecial(mon_special, 11)) {
-		var vampire_damage = hero_hp * enemy.vampire;
+		let vampire_damage = hero_hp * enemy.vampire;
 
 		// 如果有神圣盾免疫吸血等可以在这里写
 		// 也可以用hasItem和hasEquip来判定装备
@@ -695,105 +998,334 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 
 		vampire_damage = Math.floor(vampire_damage) || 0;
 		// 加到自身
-		if (enemy.add) // 如果加到自身
+		if (enemy.add)
+			// 如果加到自身
 			mon_hp += vampire_damage;
 
 		init_damage += vampire_damage;
 	}
 
-	// 每回合怪物对勇士造成的战斗伤害
-	var per_damage = mon_atk - hero_def;
+	//——第二部分:变量定义和初始赋值——
 
-	// 战斗伤害不能为负值
-	if (per_damage < 0) per_damage = 0;
+	let hero_per_damage = Math.max(hero_atk - mon_def, 0),
 
-	// 连击
-	if (core.hasSpecial(mon_special, 6)) per_damage *= (enemy.n || 2);
+		hero_per_mdamage = Math.floor((hero_spell * hero_matk / 100) * (100 - mon_mdef) / 100);
 
-	// 每回合的反击伤害;反击是按照勇士的攻击次数来计算回合
-	var counterDamage = 0;
-	if (core.hasSpecial(mon_special, 8))
-		counterDamage += Math.floor((enemy.counterAttack || core.values.counterAttack) * hero_atk);
+	let damage = 0,
+		hero_turn = 0,
+		mon_turn = 0;
+	let equipInfo = [] //回合生效的装备列表
 
-	// 先攻
-	if (core.hasSpecial(mon_special, 1)) init_damage += per_damage;
+	for (let i = 0; i < 5; i++) {
+		const a = core.plugin.equip[core.getEquip(i)]
+		if (a) equipInfo.push(a)
+	}
 
-	// 破甲
-	if (core.hasSpecial(mon_special, 7))
-		init_damage += Math.floor((enemy.breakArmor || core.values.breakArmor) * hero_def);
-
-	// 净化
-	if (core.hasSpecial(mon_special, 9))
-		init_damage += Math.floor((enemy.purify || core.values.purify) * hero_mdef);
-
-	// 勇士每回合对怪物造成的伤害
-	var hero_per_damage = Math.max(hero_atk - mon_def, 0);
-
-	// 如果没有破防,则不可战斗
-	if (hero_per_damage <= 0) return null;
-
-	// 勇士的攻击回合数;为怪物生命除以每回合伤害向上取整
-	var turn = Math.ceil(mon_hp / hero_per_damage);
-
-	// ------ 支援 ----- //
-	// 这个递归最好想明白为什么,flag:__extraTurn__是怎么用的
-	var guards = core.getFlag("__guards__" + x + "_" + y, enemyInfo.guards);
-	var guard_before_current_enemy = false; // ------ 支援怪是先打(true)还是后打(false)?
-	turn += core.getFlag("__extraTurn__", 0);
-	if (guards.length > 0) {
-		if (!guard_before_current_enemy) { // --- 先打当前怪物,记录当前回合数
-			core.setFlag("__extraTurn__", turn);
+	//处理回合条长度
+	let oneTurn = [hero_speed, mon_speed];
+	if (equipInfo.length > 0) {
+		for (let i = 0; i < equipInfo.length; i++) {
+			equipInfo[i].now = 0;
+			equipInfo[i].isAttack = false;
+			oneTurn.push(equipInfo[i].speed);
 		}
-		// 获得那些怪物组成小队战斗
-		for (var i = 0; i < guards.length; i++) {
-			var gx = guards[i][0],
-				gy = guards[i][1],
-				gid = guards[i][2];
-			// 递归计算支援怪伤害信息,这里不传x,y保证不会重复调用
-			// 这里的mdef传0,因为护盾应该只会被计算一次
-			var info = core.enemys.getDamageInfo(core.material.enemys[gid], { hp: origin_hero_hp, atk: origin_hero_atk, def: origin_hero_def, mdef: 0 });
-			if (info == null) { // 小队中任何一个怪物不可战斗,直接返回null
-				core.removeFlag("__extraTurn__");
-				return null;
+	}
+	//需要变更
+
+	const onegcd = gcd(...oneTurn) //最大公约数
+	oneTurn = lcm(...oneTurn) //单次回合长度
+	//在这里处理equip的初始位置now
+	equipInfo.forEach(v => {
+		switch (v.id) {
+		case "sword1":
+		default:
+			v.now = 0
+			break
+		}
+		v.onAttack = false
+		if (v.now === oneTurn) v.onAttack = true //增加正在攻击的标志
+	})
+	const heroinfo = { hp: hero_hp, atk: hero_atk, def: hero_def, mdef: (!hero?.mdef || hero?.mdef === 100) ? hero_mdef : hero.mdef, spell: hero_spell, mhp: Math.floor(hero_spell * hero_mhp / 100), matk: Math.floor(hero_spell * hero_matk / 100), speed: hero_speed, now: 0, isAttack: false } //勇士属性
+	const enemyinfo = { hp: mon_hp, atk: mon_atk, def: mon_def, mdef: mon_mdef, spell: mon_spell, speed: mon_speed, special: mon_special, now: 0, onAttack: false, isAttack: false } //怪物属性
+	//先攻,先攻为怪物和勇士勇士行动前怪物出第一刀
+	if (core.hasSpecial(mon_special, 1)) {
+		enemyinfo.now = oneTurn
+		enemyinfo.onAttack = true
+	}
+	const start = [core.clone(heroinfo), core.clone(enemyinfo), core.clone(equipInfo), oneTurn] //记录开始战斗时的属性并转发
+	//---第三部分:递归开始---
+
+	const heroDiffList = [],
+		enemyDiffList = [],
+		heroanimateList = [],
+		enemyanimateList = [];
+
+	let beforehp = enemyinfo.hp
+	while (
+		enemyinfo.hp > 0
+	) {
+		let onattack = false
+		const hero_diff = {},
+			enemy_diff = {},
+			hero_animate = [],
+			enemy_animate = [];
+		if (enemyinfo.onAttack) { //怪物先攻的场合
+			//这里计算怪物攻击时发生的各种变化,同时计入enemy_diff
+			let mon_damage = 0
+			let hero_damage = 0
+			//伤害计算
+			let per_damage = Math.max(enemyinfo.atk - heroinfo.def, 0),
+				per_mdamage = Math.floor(enemyinfo.spell * (100 - heroinfo.mdef) / 100);
+
+			//这里记录伤害触发后的属性变化和动画,同时计入diff、damage(不要在此直接修改heroinfo和enemyinfo)
+			if (core.hasSpecial(mon_special, 6)) {
+				hero_damage += per_damage * enemy.n + per_mdamage * enemy.n
+			} else {
+				hero_damage += per_damage + per_mdamage
 			}
-			// 已经进行的回合数
-			core.setFlag("__extraTurn__", info.turn);
-			init_damage += info.damage;
+			let animate = core.plugin.enemyanimate[enemy.id] ?? "jianji2"
+			//这里可通过if更改默认的怪物攻击特效
+			hero_animate.push(animate) //勇士身上绘制jianji2动画
+			if (heroinfo.mhp + (hero_diff.mhp ?? 0) - hero_damage >= 0) {
+				hero_diff.mhp = (hero_diff.mhp ?? 0) - hero_damage
+				hero_damage = 0
+				hero_diff.hp = (hero_diff.hp ?? 0) - hero_damage
+			} else {
+				hero_damage -= heroinfo.mhp + (hero_diff.mhp ?? 0)
+				hero_diff.mhp = (hero_diff.mhp ?? 0) - heroinfo.mhp - (hero_diff.mhp ?? 0)
+
+				hero_diff.hp = (hero_diff.hp ?? 0) - hero_damage
+			}
+
+			damage += hero_damage
+			enemyinfo.onAttack = false
+			enemyinfo.now = 0
+			onattack = true
+
 		}
-		if (guard_before_current_enemy) { // --- 先打支援怪物,增加当前回合数
-			turn += core.getFlag("__extraTurn__", 0);
+		equipInfo.forEach(v => {
+			if (v.onAttack) {
+				let mon_damage = 0
+				let hero_damage = 0
+				//这里写生效装备的技能效果,同时对双方属性的修改计入diff(不要在此直接修改heroinfo和enemyinfo)
+				let animate = core.plugin.equipanimate[v.id] ?? "jianji2"
+				//这里可通过if更改默认的道具特效
+				enemy_animate.push(animate) //勇士身上绘制动画
+
+				v.now = 0
+				v.onAttack = false
+				onattack = true
+			}
+		})
+		if (onattack) { //先手处理完毕后的数据处理
+			heroDiffList.push(hero_diff)
+			enemyDiffList.push(enemy_diff)
+			heroanimateList.push(hero_animate)
+			enemyanimateList.push(enemy_animate)
+			//处理属性变化
+			for (let v in hero_diff) {
+				heroinfo[v] += hero_diff[v]
+			}
+			for (let v in enemy_diff) {
+				enemyinfo[v] += enemy_diff[v]
+			}
+			continue //进入下一循环
 		}
+		heroinfo.now += heroinfo.speed
+		enemyinfo.now += enemyinfo.speed
+		equipInfo.forEach(v => {
+			v.now += v.speed
+		})
+
+		if (
+			heroinfo.now >= oneTurn
+		) {
+			//勇士攻击的回合
+			let mon_damage = 0
+			let hero_damage = 0
+			//这里计算勇士攻击时发生的各种变化
+
+			//伤害计算
+			let per_damage = Math.max(heroinfo.atk - enemyinfo.def, 0)
+			let per_mdamage = Math.max(Math.floor(heroinfo.matk * (100 - enemyinfo.mdef) / 100), 0)
+			mon_damage = per_damage + per_mdamage
+			//这里记录伤害触发后的属性变化和动画,同时计入diff(不要在此直接修改heroinfo和enemyinfo)	
+			let animate = core.plugin.heroanimate[equip0] ?? "jianji2"
+			//这里可通过if更改默认的武器攻击特效
+			enemy_animate.push(animate)
+
+
+			enemy_diff.hp = (enemy_diff.hp ?? 0) - mon_damage
+			heroinfo.now -= oneTurn
+			hero_turn++
+			onattack = true
+		}
+
+
+		if (enemyinfo.now >= oneTurn) {
+			//怪物攻击的回合
+			let mon_damage = 0
+			let hero_damage = 0
+			//伤害计算
+			let per_damage = Math.max(enemyinfo.atk - heroinfo.def, 0),
+				per_mdamage = Math.floor(enemyinfo.spell * (100 - heroinfo.mdef) / 100);
+
+			//这里记录伤害触发后的属性变化和动画,同时计入diff、damage(不要在此直接修改heroinfo和enemyinfo)
+			if (core.hasSpecial(mon_special, 6)) {
+				hero_damage += per_damage * enemy.n + per_mdamage * enemy.n
+			} else {
+				hero_damage += per_damage + per_mdamage
+			}
+			let animate = core.plugin.enemyanimate[enemy.id] ?? "jianji2"
+			//这里可通过if更改默认的怪物攻击特效
+			hero_animate.push(animate) //勇士身上绘制sword动画
+			if (heroinfo.mhp + (hero_diff.mhp ?? 0) - hero_damage >= 0) {
+				hero_diff.mhp = (hero_diff.mhp ?? 0) - hero_damage
+				hero_damage = 0
+				hero_diff.hp = (hero_diff.hp ?? 0) - hero_damage
+			} else {
+				hero_damage -= heroinfo.mhp + (hero_diff.mhp ?? 0)
+				hero_diff.mhp = (hero_diff.mhp ?? 0) - heroinfo.mhp - (hero_diff.mhp ?? 0)
+
+				hero_diff.hp = (hero_diff.hp ?? 0) - hero_damage
+			}
+
+			damage += hero_damage
+			enemyinfo.now -= oneTurn
+			mon_turn++
+			onattack = true
+		}
+		equipInfo.forEach(v => {
+			if (v.now >= oneTurn) {
+				let mon_damage = 0
+				let hero_damage = 0
+				//这里写生效装备的技能效果,同时对双方属性的修改计入diff(不要在此直接修改heroinfo和enemyinfo)
+				let animate = core.plugin.equipanimate[v.id] ?? "jianji2"
+				//这里可通过if更改默认的道具特效
+				enemy_animate.push(animate) //怪物身上绘制动画
+
+				v.now -= oneTurn
+				onattack = true
+			}
+
+		})
+		if (onattack) {
+			//处理完毕后的数据处理
+			heroDiffList.push(hero_diff)
+			enemyDiffList.push(enemy_diff)
+			heroanimateList.push(hero_animate)
+			enemyanimateList.push(enemy_animate)
+			//处理属性变化
+			for (let v in hero_diff) {
+				heroinfo[v] += hero_diff[v]
+			}
+			for (let v in enemy_diff) {
+				enemyinfo[v] += enemy_diff[v]
+			}
+
+			//出手50回合怪物生命未降低直接判负,避免死循环
+			if (hero_turn === 50) {
+
+				if (enemyinfo.hp >= beforehp) {
+					return null
+				}
+			}
+		}
+
 	}
-	core.removeFlag("__extraTurn__");
+
+	//下面这些还没修改,原有技能除先攻、连击外暂时全部移除,所有技能需要在上方的模拟循环中做修正
+	/*
+		// 每回合的反击伤害;反击是按照勇士的攻击次数来计算回合
+		let counterDamage = 0;
+		if (core.hasSpecial(mon_special, 8))
+			counterDamage += Math.floor(
+				(enemy.counterAttack || core.values.counterAttack) * hero_atk
+			);
+
+
+
+		// 净化
+		if (core.hasSpecial(mon_special, 9))
+			init_damage += Math.floor(
+				(enemy.purify || core.values.purify) * hero_mdef
+			);
+		//上面这些还没修改
+		//勇士护盾计算
+		let barrier = Math.floor(hero_spell * hero_mhp / 100);
+
+
+		// ------ 支援 ----- //
+		// 这个递归最好想明白为什么,flag:__extraTurn__是怎么用的
+		/*var guards = core.getFlag("__guards__" + x + "_" + y, enemyInfo.guards);
+		var guard_before_current_enemy = false; // ------ 支援怪是先打(true)还是后打(false)?
+		turn += core.getFlag("__extraTurn__", 0);
+		if (guards.length > 0) {
+			if (!guard_before_current_enemy) { // --- 先打当前怪物,记录当前回合数
+				core.setFlag("__extraTurn__", turn);
+			}
+			// 获得那些怪物组成小队战斗
+			for (var i = 0; i < guards.length; i++) {
+				var gx = guards[i][0],
+					gy = guards[i][1],
+					gid = guards[i][2];
+				// 递归计算支援怪伤害信息,这里不传x,y保证不会重复调用
+				// 这里的mdef传0,因为护盾应该只会被计算一次
+				var info = core.enemys.getDamageInfo(core.material.enemys[gid], { hp: origin_hero_hp, atk: origin_hero_atk, def: origin_hero_def, mdef: 0 });
+				if (info == null) { // 小队中任何一个怪物不可战斗,直接返回null
+					core.removeFlag("__extraTurn__");
+					return null;
+				}
+				// 已经进行的回合数
+				core.setFlag("__extraTurn__", info.turn);
+				init_damage += info.damage;
+			}
+			if (guard_before_current_enemy) { // --- 先打支援怪物,增加当前回合数
+				turn += core.getFlag("__extraTurn__", 0);
+			}
+		}
+		core.removeFlag("__extraTurn__");*/
 	// ------ 支援END ------ //
+	/*
+		// 最终伤害:初始伤害 + 怪物对勇士造成的伤害 + 反击伤害
+		damage += init_damage + hero_turn * counterDamage;
+		// 再扣去护盾
+		damage -= barrier;
 
-	// 最终伤害:初始伤害 + 怪物对勇士造成的伤害 + 反击伤害
-	var damage = init_damage + (turn - 1) * per_damage + turn * counterDamage;
-	// 再扣去护盾
-	damage -= hero_mdef;
-
-	// 检查是否允许负伤
-	if (!core.flags.enableNegativeDamage)
-		damage = Math.max(0, damage);
-
-	// 最后处理仇恨和固伤(因为这两个不能被护盾减伤)
-	if (core.hasSpecial(mon_special, 17)) { // 仇恨
-		damage += core.getFlag('hatred', 0);
-	}
-	if (core.hasSpecial(mon_special, 22)) { // 固伤
-		damage += enemy.damage || 0;
-	}
+		// 检查是否允许负伤
+		if (!core.flags.enableNegativeDamage) damage = Math.max(0, damage);
 
+		// 最后处理仇恨和固伤(因为这两个不能被护盾减伤)
+		if (core.hasSpecial(mon_special, 17)) {
+			// 仇恨
+			damage += core.getFlag("hatred", 0);
+		}
+		if (core.hasSpecial(mon_special, 22)) {
+			// 固伤
+			damage += enemy.damage2 || 0;
+		}
+	*/
 	return {
-		"mon_hp": Math.floor(mon_hp),
-		"mon_atk": Math.floor(mon_atk),
-		"mon_def": Math.floor(mon_def),
-		"init_damage": Math.floor(init_damage),
-		"per_damage": Math.floor(per_damage),
-		"hero_per_damage": Math.floor(hero_per_damage),
-		"turn": Math.floor(turn),
-		"damage": Math.floor(damage)
+		start: start,
+		mon_hp: Math.floor(mon_hp),
+		mon_atk: Math.floor(mon_atk),
+		mon_def: Math.floor(mon_def),
+		mon_mdef: Math.floor(mon_mdef),
+		mon_speed: Math.floor(mon_speed),
+		heroDiffList: heroDiffList,
+		enemyDiffList: enemyDiffList,
+		heroanimateList: heroanimateList,
+		enemyanimateList: enemyanimateList,
+		hero_turn: Math.floor(hero_turn),
+		mon_turn: Math.floor(mon_turn),
+		damage: Math.floor(damage)
 	};
+	/*TODO:怪物手册的修改(需要修改这里return的内容以及一些战后判断)
+	1. 显示怪物是魔攻还是物攻(在怪物名字上做颜色变化,物攻是黄色,魔攻是蓝色)
+	2. 一防减伤是物防还是魔防(由怪物是物攻还是魔攻来转换)
+	3. 特殊战斗的怪物,在怪物手册里“伤害”写为“特殊战”*/
+	/*TODO:怪物和勇士同时跑条到终点时,谁先出手的逻辑确定
+	 怪物、勇士和装备同时跑条时的计算*/
 }
     },
     "actions": {
@@ -807,7 +1339,8 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	if (core.isMoving()) return;
 
 	// 商店长按时忽略
-	if (core.status.onShopLongDown) return core.status.onShopLongDown = false;
+	if (core.status.onShopLongDown)
+		return (core.status.onShopLongDown = false);
 
 	// Alt+0~9,快捷换上套装
 	if (altKey && keyCode >= 48 && keyCode <= 57) {
@@ -824,7 +1357,9 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		core.openBook(true);
 		break;
 	case 71: // G:使用楼传器
-		core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
+		flags.canMoveFloor = core.canMoveFloor();
+		core.useItem('fly', true)
+		core.status.route.push("key:71");
 		break;
 	case 65: // A:读取自动存档(回退)
 		core.doSL("autoSave", "load");
@@ -882,25 +1417,37 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		core.actions._clickGameInfo_openComments();
 		break;
 	case 49: // 快捷键1: 破
-		if (core.hasItem('pickaxe')) {
+		if (core.hasItem("pickaxe")) {
 			core.status.route.push("key:49"); // 将按键记在录像中
-			core.useItem('pickaxe', true); // 第二个参数true代表该次使用道具是被按键触发的,使用过程不计入录像
+			core.useItem("pickaxe", true); // 第二个参数true代表该次使用道具是被按键触发的,使用过程不计入录像
 		}
 		break;
 	case 50: // 快捷键2: 炸
-		if (core.hasItem('bomb')) {
+		if (core.hasItem("bomb")) {
 			core.status.route.push("key:50"); // 将按键记在录像中
-			core.useItem('bomb', true); // 第二个参数true代表该次使用道具是被按键触发的,使用过程不计入录像
+			core.useItem("bomb", true); // 第二个参数true代表该次使用道具是被按键触发的,使用过程不计入录像
 		}
 		break;
 	case 51: // 快捷键3: 飞
-		if (core.hasItem('centerFly')) {
+		if (core.hasItem("centerFly")) {
 			core.ui._drawCenterFly();
 		}
 		break;
 	case 52: // 快捷键4:破冰/冰冻/地震/上下楼器/... 其他道具依次判断
 	{
-		var list = ["icePickaxe", "freezeBadge", "earthquake", "upFly", "downFly", "jumpShoes", "lifeWand", "poisonWine", "weakWine", "curseWine", "superWine"];
+		var list = [
+			"icePickaxe",
+			"freezeBadge",
+			"earthquake",
+			"upFly",
+			"downFly",
+			"jumpShoes",
+			"lifeWand",
+			"poisonWine",
+			"weakWine",
+			"curseWine",
+			"superWine",
+		];
 		for (var i = 0; i < list.length; i++) {
 			var itemId = list[i];
 			if (core.canUseItem(itemId)) {
@@ -926,9 +1473,9 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		break;
 	case 70: // F:开启技能“二倍斩”
 		// 检测是否拥有“二倍斩”这个技能道具
-		if (core.hasItem('skill1')) {
+		if (core.hasItem("skill1")) {
 			core.status.route.push("key:70");
-			core.useItem('skill1', true);
+			core.useItem("skill1", true);
 		}
 		break;
 		// 在这里可以任意新增或编辑已有的快捷键内容
@@ -946,33 +1493,32 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 					break;
 				*/
 	}
-
 },
         "onStatusBarClick": function (px, py, vertical) {
-			// 点击状态栏时触发的事件,仅在自绘状态栏开启时生效
-			// px和py为点击的像素坐标
-			// vertical为录像播放过程中的横竖屏信息
-			// 
-			// 横屏模式下状态栏的画布大小是 129*416 (开启拓展装备栏后是 129*457)
-			// 竖屏模式下状态栏的画布大小是 416*(32*rows+9) 其中rows为状态栏行数,即全塔属性中statusCanvasRowsOnMobile值
-			// 可以使用 _isVertical() 来判定当前是否是竖屏模式
+      // 点击状态栏时触发的事件,仅在自绘状态栏开启时生效
+      // px和py为点击的像素坐标
+      // vertical为录像播放过程中的横竖屏信息
+      //
+      // 横屏模式下状态栏的画布大小是 129*416 (开启拓展装备栏后是 129*457)
+      // 竖屏模式下状态栏的画布大小是 416*(32*rows+9) 其中rows为状态栏行数,即全塔属性中statusCanvasRowsOnMobile值
+      // 可以使用 _isVertical() 来判定当前是否是竖屏模式
 
-			// 判定当前是否是竖屏模式。录像播放过程中可能会记录当时的横竖屏信息以覆盖。
-			var _isVertical = function () {
-				if (core.isReplaying() && vertical != null) return vertical;
-				return core.domStyle.isVertical;
-			}
+      // 判定当前是否是竖屏模式。录像播放过程中可能会记录当时的横竖屏信息以覆盖。
+      var _isVertical = function () {
+        if (core.isReplaying() && vertical != null) return vertical;
+        return core.domStyle.isVertical;
+      };
 
-			// 如果正在执行事件,则忽略
-			if (core.status.lockControl) return;
-			// 如果当前正在行走,则忽略;也可以使用 core.waitHeroToStop(callback) 来停止行走再回调执行脚本
-			if (core.isMoving()) return;
+      // 如果正在执行事件,则忽略
+      if (core.status.lockControl) return;
+      // 如果当前正在行走,则忽略;也可以使用 core.waitHeroToStop(callback) 来停止行走再回调执行脚本
+      if (core.isMoving()) return;
 
-			// 判定px和py来执行自己的脚本内容.... 注意横竖屏
-			// console.log("onStatusBarClick: ", px, py, _isVertical());
+      // 判定px和py来执行自己的脚本内容.... 注意横竖屏
+      // console.log("onStatusBarClick: ", px, py, _isVertical());
 
-			// 样例一:点击某个区域后使用一个道具
-			/*
+      // 样例一:点击某个区域后使用一个道具
+      /*
 			if (core.hasItem("pickaxe")) {
 				if (_isVertical()) {
 					// 竖屏模式下
@@ -988,8 +1534,8 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 			}
 			*/
 
-			// 样例二:点击某个区域后执行一段公共事件或脚本
-			/*
+      // 样例二:点击某个区域后执行一段公共事件或脚本
+      /*
 			if (core.hasFlag("xxx")) {
 				if (_isVertical()) {
 					// 竖屏模式下
@@ -1014,8 +1560,7 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 				}
 			}
 			*/
-
-		}
+    }
     },
     "control": {
         "saveData": function () {
@@ -1027,31 +1572,66 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		if (!core.same(core.values[key], core.data.values[key]))
 			values[key] = core.clone(core.values[key]);
 	}
-
+	let cg = {}
+	if (core.getFlag("_cgText")) {
+		cg.index = core.ui.cgText.index
+		cg.head = core.ui.cgText.head
+		cg.name = core.ui.cgText.name
+		cg.text = core.ui.cgText.text
+		cg.bodyList = core.clone(core.ui.cgText.bodyList)
+		cg.nobg = core.ui.cgText.nobg
+		cg.image = core.ui.cgText.image
+		cg.time = core.ui.cgText.time
+		cg.WindowSkin = core.ui.cgText.WindowSkin
+		cg.sound = core.ui.cgText.sound
+		cg.beforeSound = core.ui.cgText.beforeSound
+		cg.wait = core.ui.cgText.wait
+		cg.memory = core.ui.cgText.memory
+		cg.textList = core.ui.cgText.textList
+		cg.page = core.ui.cgText.page
+		cg.overpage = core.ui.cgText.overpage
+		cg.log = core.ui.cgText.log
+		cg.index = core.ui.cgText.index
+	}
 	// 要存档的内容
 	var data = {
-		'floorId': core.status.floorId,
-		'hero': core.clone(core.status.hero),
-		'hard': core.status.hard,
-		'maps': core.clone(core.maps.saveMap()),
-		'route': core.encodeRoute(core.status.route),
-		'values': values,
-		'version': core.firstData.version,
-		'guid': core.getGuid(),
-		"time": new Date().getTime()
+		floorId: core.status.floorId,
+		hero: core.clone(core.status.hero),
+		hard: core.status.hard,
+		maps: core.clone(core.maps.saveMap()),
+		route: core.encodeRoute(core.status.route),
+		values: values,
+		version: core.firstData.version,
+		guid: core.getGuid(),
+		time: new Date().getTime(),
+		cg: cg,
+		animateObjs: core.status.animateObjs.filter(v => v.loop),
+		playing: [...core.plugin.playing].filter(v => v.loop)
 	};
 	return data;
 },
         "loadData": function (data, callback) {
 	// 读档操作;从存储中读取了内容后的行为
-	const play = core.status.played
+	const play = core.status.played;
 	// 重置游戏和路线
-	core.resetGame(data.hero, data.hard, data.floorId, core.maps.loadMap(data.maps, null, data.hero.flags), data.values);
+	core.resetGame(
+		data.hero,
+		data.hard,
+		data.floorId,
+		core.maps.loadMap(data.maps, null, data.hero.flags),
+		data.values
+	);
 	core.status.route = core.decodeRoute(data.route);
 	core.control._bindRoutePush();
 	// 文字属性,全局属性
-	core.status.textAttribute = core.getFlag('textAttribute', core.status.textAttribute);
-	var toAttribute = core.getFlag('globalAttribute', core.status.globalAttribute);
+	core.status.textAttribute = core.getFlag(
+		"textAttribute",
+		core.status.textAttribute
+	);
+	var toAttribute = core.getFlag(
+		"globalAttribute",
+		core.status.globalAttribute
+	);
 	if (!core.same(toAttribute, core.status.globalAttribute)) {
 		core.status.globalAttribute = toAttribute;
 		core.resize();
@@ -1063,36 +1643,46 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	icon = core.getMappedName(icon);
 	if (core.material.images.images[icon]) {
 		core.material.images.hero = core.material.images.images[icon];
-		core.material.icons.hero.width = core.material.images.images[icon].width / 4;
-		core.material.icons.hero.height = core.material.images.images[icon].height / 4;
+		core.material.icons.hero.width =
+			core.material.images.images[icon].width / 4;
+		core.material.icons.hero.height =
+			core.material.images.images[icon].height / 4;
 	}
-	core.setFlag('__fromLoad__', true);
+	core.setFlag("__fromLoad__", true);
 
 	// TODO:增加自己的一些读档处理
-	core.ui.statusBar.clearItemInfo()
+	core.ui.statusBar.clearItemInfo();
 	core.ui.statusBar.update();
-	core.plugin.playing.clear()
+	core.status.animateObjs = data.animateObjs
+	core.plugin.playing = new Set(data.playing)
+	if (core.getFlag("_cgText")) {
+		core.setFlag("_cgText", false)
+		for (let v in data.cg) {
+
+			core.ui.cgText[v] = data.cg[v]
+		}
+		core.drawbackground(core.ui.cgText.image, core.ui.cgText.memory)
+	}
 	// 切换到对应的楼层
 	core.changeFloor(data.floorId, null, data.hero.loc, 0, function () {
 		// TODO:可以在这里设置读档后播放BGM
-		if (core.hasFlag("__bgm__")) { // 持续播放
+		if (core.hasFlag("__bgm__")) {
+			// 持续播放
 			core.playBgm(core.getFlag("__bgm__"));
 		}
 
-		core.removeFlag('__fromLoad__');
-		if (!play) core.insertCommonEvent('强制横屏')
+		core.removeFlag("__fromLoad__");
+		if (!play) core.insertCommonEvent("强制横屏");
 		if (callback) callback();
 	});
-	if (play) core.doAction()
-
-
+	if (play) core.doAction();
 },
         "getStatusLabel": function (name) {
 	// 返回某个状态英文名的对应中文标签,如atk -> 攻击,def -> 防御等。
 	// 请注意此项仅影响 libs/ 下的内容(如绘制怪物手册、数据统计等)
 	// 自行定义的(比如获得道具效果)中用到的“攻击+3”等需要自己去对应地方修改
 
-	return {
+	return ({
 		name: "名称",
 		lv: "等级",
 		hpmax: "生命上限",
@@ -1101,72 +1691,76 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		mana: "魔力",
 		atk: "攻击",
 		def: "防御",
-		mdef: "法强",
+		spell: "法强",
+		matk: "魔攻比例",
+		mhp: "护盾比例",
+		mdef: "法抗",
 		speed: "速度",
 		money: "金币",
 		exp: "经验",
 		point: "加点",
 		steps: "步数",
-	} [name] || name;
+	} [name] || name);
 },
         "triggerDebuff": function (action, type) {
-			// 毒衰咒效果的获得与解除
-			// action:获得还是解除;'get'表示获得,'remove'表示解除
-			// type:一个数组表示获得了哪些毒衰咒效果;poison, weak,curse
-			if (!(type instanceof Array)) type = [type];
+      // 毒衰咒效果的获得与解除
+      // action:获得还是解除;'get'表示获得,'remove'表示解除
+      // type:一个数组表示获得了哪些毒衰咒效果;poison, weak,curse
+      if (!(type instanceof Array)) type = [type];
 
-			if (action == 'get') {
-				if (core.inArray(type, 'poison') && !core.hasFlag("poison")) {
-					// 获得毒效果
-					core.setFlag('poison', true);
-				}
-				if (core.inArray(type, 'weak') && !core.hasFlag('weak')) {
-					// 获得衰效果
-					core.setFlag('weak', true);
-					if (core.values.weakValue >= 1) {
-						// >=1,直接扣数值
-						core.addStatus('atk', -core.values.weakValue);
-						core.addStatus('def', -core.values.weakValue);
-					} else {
-						// <1,扣比例
-						core.addBuff('atk', -core.values.weakValue);
-						core.addBuff('def', -core.values.weakValue);
-					}
-				}
-				if (core.inArray(type, 'curse') && !core.hasFlag('curse')) {
-					// 获得咒效果
-					core.setFlag('curse', true);
-				}
-			} else if (action == 'remove') {
-				var success = false;
-				if (core.inArray(type, "poison") && core.hasFlag("poison")) {
-					success = true;
-					// 移除毒效果
-					core.setFlag("poison", false);
-				}
-				if (core.inArray(type, "weak") && core.hasFlag("weak")) {
-					success = true;
-					// 移除衰效果
-					core.setFlag("weak", false);
-					if (core.values.weakValue >= 1) {
-						// >=1,直接扣数值
-						core.addStatus('atk', core.values.weakValue);
-						core.addStatus('def', core.values.weakValue);
-					} else {
-						// <1,扣比例
-						core.addBuff('atk', core.values.weakValue);
-						core.addBuff('def', core.values.weakValue);
-					}
-				}
-				if (core.inArray(type, "curse") && core.hasFlag("curse")) {
-					success = true;
-					// 移除咒效果
-					core.setFlag("curse", false);
-				}
-				if (success) core.playSound('回血');
-			}
-		},
+      if (action == "get") {
+        if (core.inArray(type, "poison") && !core.hasFlag("poison")) {
+          // 获得毒效果
+          core.setFlag("poison", true);
+        }
+        if (core.inArray(type, "weak") && !core.hasFlag("weak")) {
+          // 获得衰效果
+          core.setFlag("weak", true);
+          if (core.values.weakValue >= 1) {
+            // >=1,直接扣数值
+            core.addStatus("atk", -core.values.weakValue);
+            core.addStatus("def", -core.values.weakValue);
+          } else {
+            // <1,扣比例
+            core.addBuff("atk", -core.values.weakValue);
+            core.addBuff("def", -core.values.weakValue);
+          }
+        }
+        if (core.inArray(type, "curse") && !core.hasFlag("curse")) {
+          // 获得咒效果
+          core.setFlag("curse", true);
+        }
+      } else if (action == "remove") {
+        var success = false;
+        if (core.inArray(type, "poison") && core.hasFlag("poison")) {
+          success = true;
+          // 移除毒效果
+          core.setFlag("poison", false);
+        }
+        if (core.inArray(type, "weak") && core.hasFlag("weak")) {
+          success = true;
+          // 移除衰效果
+          core.setFlag("weak", false);
+          if (core.values.weakValue >= 1) {
+            // >=1,直接扣数值
+            core.addStatus("atk", core.values.weakValue);
+            core.addStatus("def", core.values.weakValue);
+          } else {
+            // <1,扣比例
+            core.addBuff("atk", core.values.weakValue);
+            core.addBuff("def", core.values.weakValue);
+          }
+        }
+        if (core.inArray(type, "curse") && core.hasFlag("curse")) {
+          success = true;
+          // 移除咒效果
+          core.setFlag("curse", false);
+        }
+        if (success) core.playSound("回血");
+      }
+    },
         "updateStatusBar": function () {
+	if (!core.control.noAutoEvents) core.checkAutoEvents()
 	// 更新状态栏
 	core.ui.statusBar.update();
 	// 更新阻激夹域的伤害值
@@ -1175,216 +1769,257 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	core.updateDamage();
 },
         "updateCheckBlock": function (floorId) {
-			// 领域、夹击、阻击等的伤害值计算
-			floorId = floorId || core.status.floorId;
-			if (!floorId || !core.status.maps) return;
+      // 领域、夹击、阻击等的伤害值计算
+      floorId = floorId || core.status.floorId;
+      if (!floorId || !core.status.maps) return;
 
-			var width = core.floors[floorId].width,
-				height = core.floors[floorId].height;
-			var blocks = core.getMapBlocksObj(floorId);
+      var width = core.floors[floorId].width,
+        height = core.floors[floorId].height;
+      var blocks = core.getMapBlocksObj(floorId);
 
-			var damage = {}, // 每个点的伤害值
-				type = {}, // 每个点的伤害类型
-				repulse = {}, // 每个点的阻击怪信息
-				ambush = {}; // 每个点的捕捉信息
-			var betweenAttackLocs = {}; // 所有可能的夹击点
-			var needCache = false;
-			var canGoDeadZone = core.flags.canGoDeadZone;
-			core.flags.canGoDeadZone = true;
+      var damage = {}, // 每个点的伤害值
+        type = {}, // 每个点的伤害类型
+        repulse = {}, // 每个点的阻击怪信息
+        ambush = {}; // 每个点的捕捉信息
+      var betweenAttackLocs = {}; // 所有可能的夹击点
+      var needCache = false;
+      var canGoDeadZone = core.flags.canGoDeadZone;
+      core.flags.canGoDeadZone = true;
 
-			// 计算血网和领域、阻击、激光的伤害,计算捕捉信息
-			for (var loc in blocks) {
-				var block = blocks[loc],
-					x = block.x,
-					y = block.y,
-					id = block.event.id,
-					enemy = core.material.enemys[id];
-				if (block.disable) continue;
+      // 计算血网和领域、阻击、激光的伤害,计算捕捉信息
+      for (var loc in blocks) {
+        var block = blocks[loc],
+          x = block.x,
+          y = block.y,
+          id = block.event.id,
+          enemy = core.material.enemys[id];
+        if (block.disable) continue;
 
-				type[loc] = type[loc] || {};
+        type[loc] = type[loc] || {};
 
-				// 血网
-				// 如需调用当前楼层的ratio可使用  core.status.maps[floorId].ratio
-				if (id == 'lavaNet' && !core.hasItem('amulet')) {
-					damage[loc] = (damage[loc] || 0) + core.values.lavaDamage;
-					type[loc][(block.event.name || "血网") + "伤害"] = true;
-				}
+        // 血网
+        // 如需调用当前楼层的ratio可使用  core.status.maps[floorId].ratio
+        if (id == "lavaNet" && !core.hasItem("amulet")) {
+          damage[loc] = (damage[loc] || 0) + core.values.lavaDamage;
+          type[loc][(block.event.name || "血网") + "伤害"] = true;
+        }
 
-				// 领域
-				// 如果要防止领域伤害,可以直接简单的将 flag:no_zone 设为true
-				if (enemy && core.hasSpecial(enemy.special, 15) && !core.hasFlag('no_zone')) {
-					// 领域范围,默认为1
-					var range = enemy.range || 1;
-					// 是否是九宫格领域
-					var zoneSquare = false;
-					if (enemy.zoneSquare != null) zoneSquare = enemy.zoneSquare;
-					// 在范围内进行搜索,增加领域伤害值
-					for (var dx = -range; dx <= range; dx++) {
-						for (var dy = -range; dy <= range; dy++) {
-							if (dx == 0 && dy == 0) continue;
-							var nx = x + dx,
-								ny = y + dy,
-								currloc = nx + "," + ny;
-							if (nx < 0 || nx >= width || ny < 0 || ny >= height) continue;
-							// 如果是十字领域,则还需要满足 |dx|+|dy|<=range
-							if (!zoneSquare && Math.abs(dx) + Math.abs(dy) > range) continue;
-							damage[currloc] = (damage[currloc] || 0) + (enemy.zone || 0);
-							type[currloc] = type[currloc] || {};
-							type[currloc]["领域伤害"] = true;
-						}
-					}
-				}
+        // 领域
+        // 如果要防止领域伤害,可以直接简单的将 flag:no_zone 设为true
+        if (
+          enemy &&
+          core.hasSpecial(enemy.special, 15) &&
+          !core.hasFlag("no_zone")
+        ) {
+          // 领域范围,默认为1
+          var range = enemy.range || 1;
+          // 是否是九宫格领域
+          var zoneSquare = false;
+          if (enemy.zoneSquare != null) zoneSquare = enemy.zoneSquare;
+          // 在范围内进行搜索,增加领域伤害值
+          for (var dx = -range; dx <= range; dx++) {
+            for (var dy = -range; dy <= range; dy++) {
+              if (dx == 0 && dy == 0) continue;
+              var nx = x + dx,
+                ny = y + dy,
+                currloc = nx + "," + ny;
+              if (nx < 0 || nx >= width || ny < 0 || ny >= height) continue;
+              // 如果是十字领域,则还需要满足 |dx|+|dy|<=range
+              if (!zoneSquare && Math.abs(dx) + Math.abs(dy) > range) continue;
+              damage[currloc] = (damage[currloc] || 0) + (enemy.zone || 0);
+              type[currloc] = type[currloc] || {};
+              type[currloc]["领域伤害"] = true;
+            }
+          }
+        }
 
-				// 阻击
-				// 如果要防止阻击伤害,可以直接简单的将 flag:no_repulse 设为true
-				if (enemy && core.hasSpecial(enemy.special, 18) && !core.hasFlag('no_repulse')) {
-					var scan = enemy.zoneSquare ? core.utils.scan2 : core.utils.scan;
-					for (var dir in scan) {
-						var nx = x + scan[dir].x,
-							ny = y + scan[dir].y,
-							currloc = nx + "," + ny;
-						if (nx < 0 || nx >= width || ny < 0 || ny >= height) continue;
-						damage[currloc] = (damage[currloc] || 0) + (enemy.repulse || 0);
-						type[currloc] = type[currloc] || {};
-						type[currloc]["阻击伤害"] = true;
+        // 阻击
+        // 如果要防止阻击伤害,可以直接简单的将 flag:no_repulse 设为true
+        if (
+          enemy &&
+          core.hasSpecial(enemy.special, 18) &&
+          !core.hasFlag("no_repulse")
+        ) {
+          var scan = enemy.zoneSquare ? core.utils.scan2 : core.utils.scan;
+          for (var dir in scan) {
+            var nx = x + scan[dir].x,
+              ny = y + scan[dir].y,
+              currloc = nx + "," + ny;
+            if (nx < 0 || nx >= width || ny < 0 || ny >= height) continue;
+            damage[currloc] = (damage[currloc] || 0) + (enemy.repulse || 0);
+            type[currloc] = type[currloc] || {};
+            type[currloc]["阻击伤害"] = true;
 
-						var rdir = core.turnDirection(":back", dir);
-						// 检查下一个点是否存在事件(从而判定是否移动)
-						var rnx = x + scan[rdir].x,
-							rny = y + scan[rdir].y;
-						if (rnx < 0 || rnx >= width || rny < 0 || rny >= height) continue;
-						// 如需禁止阻击被推到已隐藏的事件处(如重生怪处),可将这一句的false改为true
-						if (core.getBlock(rnx, rny, floorId, false) != null) continue;
-						if (core.utils.scan[rdir] && !core.canMoveHero(x, y, rdir, floorId)) continue;
-						repulse[currloc] = (repulse[currloc] || []).concat([
-							[x, y, id, rdir]
-						]);
-					}
-				}
+            var rdir = core.turnDirection(":back", dir);
+            // 检查下一个点是否存在事件(从而判定是否移动)
+            var rnx = x + scan[rdir].x,
+              rny = y + scan[rdir].y;
+            if (rnx < 0 || rnx >= width || rny < 0 || rny >= height) continue;
+            // 如需禁止阻击被推到已隐藏的事件处(如重生怪处),可将这一句的false改为true
+            if (core.getBlock(rnx, rny, floorId, false) != null) continue;
+            if (core.utils.scan[rdir] && !core.canMoveHero(x, y, rdir, floorId))
+              continue;
+            repulse[currloc] = (repulse[currloc] || []).concat([
+              [x, y, id, rdir],
+            ]);
+          }
+        }
 
-				// 激光
-				// 如果要防止激光伤害,可以直接简单的将 flag:no_laser 设为true
-				if (enemy && core.hasSpecial(enemy.special, 24) && !core.hasFlag("no_laser")) {
-					for (var nx = 0; nx < width; nx++) {
-						var currloc = nx + "," + y;
-						if (nx != x) {
-							damage[currloc] = (damage[currloc] || 0) + (enemy.laser || 0);
-							type[currloc] = type[currloc] || {};
-							type[currloc]["激光伤害"] = true;
-						}
-					}
-					for (var ny = 0; ny < height; ny++) {
-						var currloc = x + "," + ny;
-						if (ny != y) {
-							damage[currloc] = (damage[currloc] || 0) + (enemy.laser || 0);
-							type[currloc] = type[currloc] || {};
-							type[currloc]["激光伤害"] = true;
-						}
-					}
-				}
+        // 激光
+        // 如果要防止激光伤害,可以直接简单的将 flag:no_laser 设为true
+        if (
+          enemy &&
+          core.hasSpecial(enemy.special, 24) &&
+          !core.hasFlag("no_laser")
+        ) {
+          for (var nx = 0; nx < width; nx++) {
+            var currloc = nx + "," + y;
+            if (nx != x) {
+              damage[currloc] = (damage[currloc] || 0) + (enemy.laser || 0);
+              type[currloc] = type[currloc] || {};
+              type[currloc]["激光伤害"] = true;
+            }
+          }
+          for (var ny = 0; ny < height; ny++) {
+            var currloc = x + "," + ny;
+            if (ny != y) {
+              damage[currloc] = (damage[currloc] || 0) + (enemy.laser || 0);
+              type[currloc] = type[currloc] || {};
+              type[currloc]["激光伤害"] = true;
+            }
+          }
+        }
 
-				// 捕捉
-				// 如果要防止捕捉效果,可以直接简单的将 flag:no_ambush 设为true
-				if (enemy && core.enemys.hasSpecial(enemy.special, 27) && !core.hasFlag("no_ambush")) {
-					var scan = enemy.zoneSquare ? core.utils.scan2 : core.utils.scan;
-					// 给周围格子加上【捕捉】记号
-					for (var dir in scan) {
-						var nx = x + scan[dir].x,
-							ny = y + scan[dir].y,
-							currloc = nx + "," + ny;
-						if (nx < 0 || nx >= width || ny < 0 || ny >= height || (core.utils.scan[dir] && !core.canMoveHero(x, y, dir, floorId))) continue;
-						ambush[currloc] = (ambush[currloc] || []).concat([
-							[x, y, id, dir]
-						]);
-					}
-				}
+        // 捕捉
+        // 如果要防止捕捉效果,可以直接简单的将 flag:no_ambush 设为true
+        if (
+          enemy &&
+          core.enemys.hasSpecial(enemy.special, 27) &&
+          !core.hasFlag("no_ambush")
+        ) {
+          var scan = enemy.zoneSquare ? core.utils.scan2 : core.utils.scan;
+          // 给周围格子加上【捕捉】记号
+          for (var dir in scan) {
+            var nx = x + scan[dir].x,
+              ny = y + scan[dir].y,
+              currloc = nx + "," + ny;
+            if (
+              nx < 0 ||
+              nx >= width ||
+              ny < 0 ||
+              ny >= height ||
+              (core.utils.scan[dir] && !core.canMoveHero(x, y, dir, floorId))
+            )
+              continue;
+            ambush[currloc] = (ambush[currloc] || []).concat([[x, y, id, dir]]);
+          }
+        }
 
-				// 夹击;在这里提前计算所有可能的夹击点,具体计算逻辑在下面
-				// 如果要防止夹击伤害,可以简单的将 flag:no_betweenAttack 设为true
-				if (enemy && core.enemys.hasSpecial(enemy.special, 16) && !core.hasFlag('no_betweenAttack')) {
-					for (var dir in core.utils.scan) {
-						var nx = x + core.utils.scan[dir].x,
-							ny = y + core.utils.scan[dir].y,
-							currloc = nx + "," + ny;
-						if (nx < 0 || nx >= width || ny < 0 || ny >= height) continue;
-						betweenAttackLocs[currloc] = true;
-					}
-				}
+        // 夹击;在这里提前计算所有可能的夹击点,具体计算逻辑在下面
+        // 如果要防止夹击伤害,可以简单的将 flag:no_betweenAttack 设为true
+        if (
+          enemy &&
+          core.enemys.hasSpecial(enemy.special, 16) &&
+          !core.hasFlag("no_betweenAttack")
+        ) {
+          for (var dir in core.utils.scan) {
+            var nx = x + core.utils.scan[dir].x,
+              ny = y + core.utils.scan[dir].y,
+              currloc = nx + "," + ny;
+            if (nx < 0 || nx >= width || ny < 0 || ny >= height) continue;
+            betweenAttackLocs[currloc] = true;
+          }
+        }
 
-				// 检查地图范围类技能
-				var specialFlag = core.getSpecialFlag(enemy);
-				if (specialFlag & 1) needCache = true;
-				if (core.status.event.id == 'viewMaps') needCache = true;
-				if ((core.status.event.id == 'book' || core.status.event.id == 'bool-detail') && core.status.event.ui) needCache = true;
-			}
+        // 检查地图范围类技能
+        var specialFlag = core.getSpecialFlag(enemy);
+        if (specialFlag & 1) needCache = true;
+        if (core.status.event.id == "viewMaps") needCache = true;
+        if (
+          (core.status.event.id == "book" ||
+            core.status.event.id == "bool-detail") &&
+          core.status.event.ui
+        )
+          needCache = true;
+      }
 
-			// 对每个可能的夹击点计算夹击伤害
-			for (var loc in betweenAttackLocs) {
-				var xy = loc.split(","),
-					x = parseInt(xy[0]),
-					y = parseInt(xy[1]);
-				// 夹击怪物的ID
-				var enemyId1 = null,
-					enemyId2 = null;
-				// 检查左右夹击
-				var leftBlock = blocks[(x - 1) + "," + y],
-					rightBlock = blocks[(x + 1) + "," + y];
-				var leftId = core.getFaceDownId(leftBlock),
-					rightId = core.getFaceDownId(rightBlock);
-				if (leftBlock && !leftBlock.disable && rightBlock && !rightBlock.disable && leftId == rightId) {
-					if (core.hasSpecial(leftId, 16))
-						enemyId1 = leftId;
-				}
-				// 检查上下夹击
-				var topBlock = blocks[x + "," + (y - 1)],
-					bottomBlock = blocks[x + "," + (y + 1)];
-				var topId = core.getFaceDownId(topBlock),
-					bottomId = core.getFaceDownId(bottomBlock);
-				if (topBlock && !topBlock.disable && bottomBlock && !bottomBlock.disable && topId == bottomId) {
-					if (core.hasSpecial(topId, 16))
-						enemyId2 = topId;
-				}
+      // 对每个可能的夹击点计算夹击伤害
+      for (var loc in betweenAttackLocs) {
+        var xy = loc.split(","),
+          x = parseInt(xy[0]),
+          y = parseInt(xy[1]);
+        // 夹击怪物的ID
+        var enemyId1 = null,
+          enemyId2 = null;
+        // 检查左右夹击
+        var leftBlock = blocks[x - 1 + "," + y],
+          rightBlock = blocks[x + 1 + "," + y];
+        var leftId = core.getFaceDownId(leftBlock),
+          rightId = core.getFaceDownId(rightBlock);
+        if (
+          leftBlock &&
+          !leftBlock.disable &&
+          rightBlock &&
+          !rightBlock.disable &&
+          leftId == rightId
+        ) {
+          if (core.hasSpecial(leftId, 16)) enemyId1 = leftId;
+        }
+        // 检查上下夹击
+        var topBlock = blocks[x + "," + (y - 1)],
+          bottomBlock = blocks[x + "," + (y + 1)];
+        var topId = core.getFaceDownId(topBlock),
+          bottomId = core.getFaceDownId(bottomBlock);
+        if (
+          topBlock &&
+          !topBlock.disable &&
+          bottomBlock &&
+          !bottomBlock.disable &&
+          topId == bottomId
+        ) {
+          if (core.hasSpecial(topId, 16)) enemyId2 = topId;
+        }
 
-				if (enemyId1 != null || enemyId2 != null) {
-					var leftHp = core.status.hero.hp - (damage[loc] || 0);
-					if (leftHp > 1) {
-						// 夹击伤害值
-						var value = Math.floor(leftHp / 2);
-						// 是否不超过怪物伤害值
-						if (core.flags.betweenAttackMax) {
-							var enemyDamage1 = core.getDamage(enemyId1, x, y, floorId);
-							if (enemyDamage1 != null && enemyDamage1 < value)
-								value = enemyDamage1;
-							var enemyDamage2 = core.getDamage(enemyId2, x, y, floorId);
-							if (enemyDamage2 != null && enemyDamage2 < value)
-								value = enemyDamage2;
-						}
-						if (value > 0) {
-							damage[loc] = (damage[loc] || 0) + value;
-							type[loc] = type[loc] || {};
-							type[loc]["夹击伤害"] = true;
-						}
-					}
-				}
-			}
+        if (enemyId1 != null || enemyId2 != null) {
+          var leftHp = core.status.hero.hp - (damage[loc] || 0);
+          if (leftHp > 1) {
+            // 夹击伤害值
+            var value = Math.floor(leftHp / 2);
+            // 是否不超过怪物伤害值
+            if (core.flags.betweenAttackMax) {
+              var enemyDamage1 = core.getDamage(enemyId1, x, y, floorId);
+              if (enemyDamage1 != null && enemyDamage1 < value)
+                value = enemyDamage1;
+              var enemyDamage2 = core.getDamage(enemyId2, x, y, floorId);
+              if (enemyDamage2 != null && enemyDamage2 < value)
+                value = enemyDamage2;
+            }
+            if (value > 0) {
+              damage[loc] = (damage[loc] || 0) + value;
+              type[loc] = type[loc] || {};
+              type[loc]["夹击伤害"] = true;
+            }
+          }
+        }
+      }
 
-			// 取消注释下面这一段可以让护盾抵御阻激夹域伤害
-			/*
+      // 取消注释下面这一段可以让护盾抵御阻激夹域伤害
+      /*
 			for (var loc in damage) {
 				damage[loc] = Math.max(0, damage[loc] - core.getRealStatus('mdef'));
 			}
 			*/
 
-			core.flags.canGoDeadZone = canGoDeadZone;
-			core.status.checkBlock = {
-				damage: damage,
-				type: type,
-				repulse: repulse,
-				ambush: ambush,
-				needCache: needCache,
-				cache: {} // clear cache
-			};
-		},
+      core.flags.canGoDeadZone = canGoDeadZone;
+      core.status.checkBlock = {
+        damage: damage,
+        type: type,
+        repulse: repulse,
+        ambush: ambush,
+        needCache: needCache,
+        cache: {}, // clear cache
+      };
+    },
         "moveOneStep": function (callback) {
 	// 勇士每走一步后执行的操作。callback为行走完毕后的回调
 	// 这个函数执行在“刚走完”的时候,即还没有检查该点的事件和领域伤害等。
@@ -1399,7 +2034,7 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	core.updateFollowers();
 	core.drawHero();
 	// 检查中毒状态的扣血和死亡
-	if (core.hasFlag('poison')) {
+	if (core.hasFlag("poison")) {
 		core.status.hero.statistics.poisonDamage += core.values.poisonDamage;
 		core.status.hero.hp -= core.values.poisonDamage;
 		if (core.status.hero.hp <= 0) {
@@ -1421,12 +2056,16 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 
 	// ------ 检查目标点事件 ------ //
 	// 无事件的道具(如血瓶)需要优先于阻激夹域判定
-	var nowx = core.getHeroLoc('x'),
-		nowy = core.getHeroLoc('y');
+	var nowx = core.getHeroLoc("x"),
+		nowy = core.getHeroLoc("y");
 	var block = core.getBlock(nowx, nowy);
 	var hasTrigger = false;
-	if (block != null && block.event.trigger == 'getItem' &&
-		!core.floors[core.status.floorId].afterGetItem[nowx + "," + nowy]) {
+	core.showComment(nowx, nowy)
+	if (
+		block != null &&
+		block.event.trigger == "getItem" &&
+		!core.floors[core.status.floorId].afterGetItem[nowx + "," + nowy]
+	) {
 		hasTrigger = true;
 		core.trigger(nowx, nowy, callback);
 	}
@@ -1434,13 +2073,12 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	core.checkBlock();
 
 	// 执行目标点的script和事件
-	if (!hasTrigger)
-		core.trigger(nowx, nowy, callback);
+	if (!hasTrigger) core.trigger(nowx, nowy, callback);
 
 	// 检查该点是否是滑冰
 	if (core.onSki()) {
 		// 延迟到事件最后执行,因为这之前可能有阻激夹域动画
-		core.insertAction({ "type": "moveAction" }, null, null, null, true);
+		core.insertAction({ type: "moveAction" }, null, null, null, true);
 	}
 
 	// ------ 检查目标点事件 END ------ //
@@ -1456,19 +2094,23 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	if (ignoreSteps == null) ignoreSteps = core.canMoveDirectly(x, y);
 	if (ignoreSteps >= 0) {
 		// 中毒也允许瞬移
-		if (core.hasFlag('poison')) {
+		if (core.hasFlag("poison")) {
 			var damage = ignoreSteps * core.values.poisonDamage;
 			if (damage >= core.status.hero.hp) return false;
 			core.status.hero.statistics.poisonDamage += damage;
 			core.status.hero.hp -= damage;
 		}
-
-		core.clearMap('hero');
-		if ((core.bigmap.width * 32) === (core.bigmap.height * 32) && (core.bigmap.width * 32) === core.__PIXELS__) core.addPopMove(32 * core.status.hero.loc.x + 16, 32 * core.status.hero.loc.y + 16, 32 * x + 16, 32 * y + 16)
+		core.showComment(x, y)
+		core.clearMap("hero");
+		if (
+			core.bigmap.width * 32 === core.bigmap.height * 32 &&
+			core.bigmap.width * 32 === core.__PIXELS__
+		)
+			if (core.getFlag('popmove')) core.addPopMove(32 * core.status.hero.loc.x + 16, 32 * core.status.hero.loc.y + 16, 32 * x + 16, 32 * y + 16);
 		// 获得勇士最后的朝向
 		var lastDirection = core.status.route[core.status.route.length - 1];
-		if (['left', 'right', 'up', 'down'].indexOf(lastDirection) >= 0)
-			core.setHeroLoc('direction', lastDirection);
+		if (["left", "right", "up", "down"].indexOf(lastDirection) >= 0)
+			core.setHeroLoc("direction", lastDirection);
 		// 设置坐标,并绘制
 		core.control._moveDirectyFollowers(x, y);
 		core.status.hero.loc.x = x;
@@ -1479,7 +2121,7 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 		// 统计信息
 		core.status.hero.statistics.moveDirectly++;
 		core.status.hero.statistics.ignoreSteps += ignoreSteps;
-		if (core.hasFlag('poison')) {
+		if (core.hasFlag("poison")) {
 			core.updateStatusBar(false, true);
 		}
 
@@ -1489,148 +2131,266 @@ var functions_d6ad677b_427a_4623_b50f_a445a3b0ef8a =
 	return false;
 },
         "parallelDo": function (timestamp) {
-			// 并行事件处理,可以在这里写任何需要并行处理的脚本或事件
-			// 该函数将被系统反复执行,每次执行间隔视浏览器或设备性能而定,一般约为16.6ms一次
-			// 参数timestamp为“从游戏资源加载完毕到当前函数执行时”的时间差,以毫秒为单位
+      // 并行事件处理,可以在这里写任何需要并行处理的脚本或事件
+      // 该函数将被系统反复执行,每次执行间隔视浏览器或设备性能而定,一般约为16.6ms一次
+      // 参数timestamp为“从游戏资源加载完毕到当前函数执行时”的时间差,以毫秒为单位
 
-			// 检查当前是否处于游戏开始状态
-			if (!core.isPlaying()) return;
+      // 检查当前是否处于游戏开始状态
+      if (!core.isPlaying()) return;
 
-			// 执行当前楼层的并行事件处理
-			if (core.status.floorId) {
-				try {
-					eval(core.floors[core.status.floorId].parallelDo);
-				} catch (e) {
-					console.error(e);
-				}
-			}
-		}
+      // 执行当前楼层的并行事件处理
+      if (core.status.floorId) {
+        try {
+          eval(core.floors[core.status.floorId].parallelDo);
+        } catch (e) {
+          console.error(e);
+        }
+      }
+    }
     },
     "ui": {
         "getToolboxItems": function (cls) {
-	// 获得道具栏中当前某类型道具的显示项和显示顺序
-	// cls为道具类型,只可能是 tools, constants 和 equips
-	// 返回一个数组,代表当前某类型道具的显示内容和顺序
-	// 默认按id升序排列,您可以取消下面的注释改为按名称排列
+      // 获得道具栏中当前某类型道具的显示项和显示顺序
+      // cls为道具类型,只可能是 tools, constants 和 equips
+      // 返回一个数组,代表当前某类型道具的显示内容和顺序
+      // 默认按id升序排列,您可以取消下面的注释改为按名称排列
 
-	return Object.keys(core.status.hero.items[cls] || {})
-		.filter(function (id) { return !core.material.items[id].hideInToolbox; })
-		.sort( /*function (id1, id2) { return core.material.items[id1].name <= core.material.items[id2].name ? -1 : 1 }*/ );
-},
+      return Object.keys(core.status.hero.items[cls] || {})
+        .filter(function (id) {
+          return !core.material.items[id].hideInToolbox;
+        })
+        .sort(/*function (id1, id2) { return core.material.items[id1].name <= core.material.items[id2].name ? -1 : 1 }*/);
+    },
         "drawStatusBar": function () {
 	// 这真的是人能写出来的东西?
-	var ctx, fill = function (text, x, y, style) {
-		core.ui.setFont(ctx, (/\w+/.test(text) ? 'italic ' : '') + 'bold 18px Verdana');
-		core.ui.fillBoldText(ctx, text, x, y, style);
-	};
-	if (core.flags.statusCanvas) { // 系统开关「自绘状态栏」开启
-		core.ui.clearMap(ctx = core.dom.statusCanvasCtx); // 清空状态栏
+	var ctx,
+		fill = function (text, x, y, style) {
+			core.ui.setFont(
+				ctx,
+				(/\w+/.test(text) ? "italic " : "") + "bold 18px Verdana"
+			);
+			core.ui.fillBoldText(ctx, text, x, y, style);
+		};
+	if (core.flags.statusCanvas) {
+		// 系统开关「自绘状态栏」开启
+		core.ui.clearMap((ctx = core.dom.statusCanvasCtx)); // 清空状态栏
 		core.ui.setFillStyle(ctx, core.status.globalAttribute.statusBarColor);
-		if (core.domStyle.isVertical) { // 竖屏
+		if (core.domStyle.isVertical) {
+			// 竖屏
 			core.drawImage(ctx, core.statusBar.icons.floor, 6, 6, 25, 25);
 			fill((core.status.thisMap || {}).name || "Loading", 42, 26);
 			core.drawImage(ctx, core.statusBar.icons.hp, 137, 6, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('hp')), 173, 26);
+			fill(core.formatBigNumber(core.getRealStatus("hp")), 173, 26);
 			core.drawImage(ctx, core.statusBar.icons.atk, 268, 6, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('atk')), 304, 26);
+			fill(core.formatBigNumber(core.getRealStatus("atk")), 304, 26);
 			core.drawImage(ctx, core.statusBar.icons.def, 6, 38, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('def')), 42, 58);
+			fill(core.formatBigNumber(core.getRealStatus("def")), 42, 58);
 			core.drawImage(ctx, core.statusBar.icons.mdef, 137, 38, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('mdef')), 173, 58);
+			fill(core.formatBigNumber(core.getRealStatus("mdef")), 173, 58);
 			core.drawImage(ctx, core.statusBar.icons.money, 268, 38, 25, 25);
 			fill(core.formatBigNumber(core.status.hero.money), 304, 58);
 			core.drawImage(ctx, core.statusBar.icons.exp, 6, 70, 25, 25);
 			fill(core.formatBigNumber(core.status.hero.exp), 42, 90);
-		} else if (!core.flags.hideLeftStatusBar) { // 横屏且未隐藏状态栏
+		} else if (!core.flags.hideLeftStatusBar) {
+			// 横屏且未隐藏状态栏
 			core.drawImage(ctx, core.statusBar.icons.floor, 6, 9, 25, 25);
 			fill((core.status.thisMap || {}).name || "Loading", 42, 29);
 			core.drawImage(ctx, core.statusBar.icons.hp, 6, 43, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('hp')), 42, 63);
+			fill(core.formatBigNumber(core.getRealStatus("hp")), 42, 63);
 			core.drawImage(ctx, core.statusBar.icons.atk, 6, 77, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('atk')), 42, 97);
+			fill(core.formatBigNumber(core.getRealStatus("atk")), 42, 97);
 			core.drawImage(ctx, core.statusBar.icons.def, 6, 111, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('def')), 42, 131);
+			fill(core.formatBigNumber(core.getRealStatus("def")), 42, 131);
 			core.drawImage(ctx, core.statusBar.icons.mdef, 6, 145, 25, 25);
-			fill(core.formatBigNumber(core.getRealStatus('mdef')), 42, 165);
+			fill(core.formatBigNumber(core.getRealStatus("mdef")), 42, 165);
 			core.drawImage(ctx, core.statusBar.icons.money, 6, 179, 25, 25);
 			fill(core.formatBigNumber(core.status.hero.money), 42, 199);
 			core.drawImage(ctx, core.statusBar.icons.exp, 6, 213, 25, 25);
 			fill(core.formatBigNumber(core.status.hero.exp), 42, 233);
-			fill(core.setTwoDigits(core.itemCount('yellowKey')), 11, 267, '#FFCCAA');
-			fill(core.setTwoDigits(core.itemCount('blueKey')), 46, 267, '#AAAADD');
-			fill(core.setTwoDigits(core.itemCount('redKey')), 81, 267, '#FF8888');
+			fill(
+				core.setTwoDigits(core.itemCount("yellowKey")),
+				11,
+				267,
+				"#FFCCAA"
+			);
+			fill(
+				core.setTwoDigits(core.itemCount("blueKey")),
+				46,
+				267,
+				"#AAAADD"
+			);
+			fill(core.setTwoDigits(core.itemCount("redKey")), 81, 267, "#FF8888");
 		}
-	} else if (core.flags.hideLeftStatusBar && !core.domStyle.isVertical) { // 横屏且隐藏状态栏
-		if (!core.dymCanvas['status'])
-			core.ui.createCanvas('status', 0, 0, core._PX_, core._PY_, 66); // 刚好盖过显伤层
-		core.ui.clearMap(ctx = core.dymCanvas['status']);
+	} else if (core.flags.hideLeftStatusBar && !core.domStyle.isVertical) {
+		// 横屏且隐藏状态栏
+		if (!core.dymCanvas["status"])
+			core.ui.createCanvas("status", 0, 0, core._PX_, core._PY_, 66); // 刚好盖过显伤层
+		core.ui.clearMap((ctx = core.dymCanvas["status"]));
 		core.ui.setFillStyle(ctx, core.status.globalAttribute.statusBarColor);
-		var offset = core.status.hero.loc.x - core.bigmap.offsetX / 32 >= core._HEIGHT_ ? 0 : core._PY_;
+		var offset =
+			core.status.hero.loc.x - core.bigmap.offsetX / 32 >= core._HEIGHT_ ?
+			0 :
+			core._PY_;
 		core.ui.setAlpha(ctx, 0.75);
-		core.ui.drawWindowSkin('winskin.webp', ctx, offset, 0, core._PX_ - core._PY_, core._PY_);
+		core.ui.drawWindowSkin(
+			"winskin.webp",
+			ctx,
+			offset,
+			0,
+			core._PX_ - core._PY_,
+			core._PY_
+		);
 		core.ui.setAlpha(ctx, 1);
 		core.drawImage(ctx, core.statusBar.icons.floor, 6 + offset, 9, 25, 25);
 		fill((core.status.thisMap || {}).name || "Loading", 42 + offset, 29);
 		core.drawImage(ctx, core.statusBar.icons.hp, 6 + offset, 43, 25, 25);
-		fill(core.formatBigNumber(core.getRealStatus('hp')), 42 + offset, 63);
+		fill(core.formatBigNumber(core.getRealStatus("hp")), 42 + offset, 63);
 		core.drawImage(ctx, core.statusBar.icons.atk, 6 + offset, 77, 25, 25);
-		fill(core.formatBigNumber(core.getRealStatus('atk')), 42 + offset, 97);
+		fill(core.formatBigNumber(core.getRealStatus("atk")), 42 + offset, 97);
 		core.drawImage(ctx, core.statusBar.icons.def, 6 + offset, 111, 25, 25);
-		fill(core.formatBigNumber(core.getRealStatus('def')), 42 + offset, 131);
+		fill(core.formatBigNumber(core.getRealStatus("def")), 42 + offset, 131);
 		core.drawImage(ctx, core.statusBar.icons.mdef, 6 + offset, 145, 25, 25);
-		fill(core.formatBigNumber(core.getRealStatus('mdef')), 42 + offset, 165);
-		core.drawImage(ctx, core.statusBar.icons.money, 6 + offset, 179, 25, 25);
+		fill(
+			core.formatBigNumber(core.getRealStatus("mdef")),
+			42 + offset,
+			165
+		);
+		core.drawImage(
+			ctx,
+			core.statusBar.icons.money,
+			6 + offset,
+			179,
+			25,
+			25
+		);
 		fill(core.formatBigNumber(core.status.hero.money), 42 + offset, 199);
 		core.drawImage(ctx, core.statusBar.icons.exp, 6 + offset, 213, 25, 25);
 		fill(core.formatBigNumber(core.status.hero.exp), 42 + offset, 233);
-		fill(core.setTwoDigits(core.itemCount('yellowKey')), 11 + offset, 267, '#FFCCAA');
-		fill(core.setTwoDigits(core.itemCount('blueKey')), 46 + offset, 267, '#AAAADD');
-		fill(core.setTwoDigits(core.itemCount('redKey')), 81 + offset, 267, '#FF8888');
+		fill(
+			core.setTwoDigits(core.itemCount("yellowKey")),
+			11 + offset,
+			267,
+			"#FFCCAA"
+		);
+		fill(
+			core.setTwoDigits(core.itemCount("blueKey")),
+			46 + offset,
+			267,
+			"#AAAADD"
+		);
+		fill(
+			core.setTwoDigits(core.itemCount("redKey")),
+			81 + offset,
+			267,
+			"#FF8888"
+		);
 	}
 },
         "drawStatistics": function () {
-			// 浏览地图时参与的统计项目
+      // 浏览地图时参与的统计项目
 
-			return [
-				'yellowDoor', 'blueDoor', 'redDoor', 'greenDoor', 'steelDoor',
-				'yellowKey', 'blueKey', 'redKey', 'greenKey', 'steelKey',
-				'redGem', 'blueGem', 'greenGem', 'yellowGem',
-				'redPotion', 'bluePotion', 'greenPotion', 'yellowPotion', 'superPotion',
-				'pickaxe', 'bomb', 'centerFly', 'icePickaxe', 'freezeBadge',
-				'earthquake', 'upFly', 'downFly', 'jumpShoes', 'lifeWand',
-				'poisonWine', 'weakWine', 'curseWine', 'superWine',
-				'sword1', 'sword2', 'sword3', 'sword4', 'sword5',
-				'shield1', 'shield2', 'shield3', 'shield4', 'shield5',
-				// 在这里可以增加新的ID来进行统计个数,只能增加道具ID
-			];
-		},
+      return [
+        "yellowDoor",
+        "blueDoor",
+        "redDoor",
+        "greenDoor",
+        "steelDoor",
+        "yellowKey",
+        "blueKey",
+        "redKey",
+        "greenKey",
+        "steelKey",
+        "redGem",
+        "blueGem",
+        "greenGem",
+        "yellowGem",
+        "redPotion",
+        "bluePotion",
+        "greenPotion",
+        "yellowPotion",
+        "superPotion",
+        "pickaxe",
+        "bomb",
+        "centerFly",
+        "icePickaxe",
+        "freezeBadge",
+        "earthquake",
+        "upFly",
+        "downFly",
+        "jumpShoes",
+        "lifeWand",
+        "poisonWine",
+        "weakWine",
+        "curseWine",
+        "superWine",
+        "sword1",
+        "sword2",
+        "sword3",
+        "sword4",
+        "sword5",
+        "shield1",
+        "shield2",
+        "shield3",
+        "shield4",
+        "shield5",
+        // 在这里可以增加新的ID来进行统计个数,只能增加道具ID
+      ];
+    },
         "drawAbout": function () {
-			// 绘制“关于”界面
-			core.ui.closePanel();
-			core.lockControl();
-			core.status.event.id = 'about';
+      // 绘制“关于”界面
+      core.ui.closePanel();
+      core.lockControl();
+      core.status.event.id = "about";
 
-			var left = 48,
-				top = 36,
-				right = (core._PX_ || core.__PIXELS__) - 2 * left,
-				bottom = (core._PY_ || core.__PIXELS__) - 2 * top;
+      var left = 48,
+        top = 36,
+        right = (core._PX_ || core.__PIXELS__) - 2 * left,
+        bottom = (core._PY_ || core.__PIXELS__) - 2 * top;
 
-			core.setAlpha('ui', 0.85);
-			core.fillRect('ui', left, top, right, bottom, '#000000');
-			core.setAlpha('ui', 1);
-			core.strokeRect('ui', left - 1, top - 1, right + 1, bottom + 1, '#FFFFFF', 2);
+      core.setAlpha("ui", 0.85);
+      core.fillRect("ui", left, top, right, bottom, "#000000");
+      core.setAlpha("ui", 1);
+      core.strokeRect(
+        "ui",
+        left - 1,
+        top - 1,
+        right + 1,
+        bottom + 1,
+        "#FFFFFF",
+        2
+      );
 
-			var text_start = left + 24;
+      var text_start = left + 24;
 
-			// 名称
-			core.setTextAlign('ui', 'left');
-			var globalAttribute = core.status.globalAttribute || core.initStatus.globalAttribute;
-			core.fillText('ui', "HTML5 魔塔样板", text_start, top + 35, globalAttribute.selectColor, "bold 22px " + globalAttribute.font);
-			core.fillText('ui', "版本: " + main.__VERSION__, text_start, top + 80, "#FFFFFF", "bold 17px " + globalAttribute.font);
-			core.fillText('ui', "作者: 艾之葵", text_start, top + 112);
-			core.fillText('ui', 'HTML5魔塔交流群:539113091', text_start, top + 112 + 32);
-			// TODO: 写自己的“关于”页面,每次增加32像素即可
-			core.playSound('打开界面');
-		}
+      // 名称
+      core.setTextAlign("ui", "left");
+      var globalAttribute =
+        core.status.globalAttribute || core.initStatus.globalAttribute;
+      core.fillText(
+        "ui",
+        "HTML5 魔塔样板",
+        text_start,
+        top + 35,
+        globalAttribute.selectColor,
+        "bold 22px " + globalAttribute.font
+      );
+      core.fillText(
+        "ui",
+        "版本: " + main.__VERSION__,
+        text_start,
+        top + 80,
+        "#FFFFFF",
+        "bold 17px " + globalAttribute.font
+      );
+      core.fillText("ui", "作者: 艾之葵", text_start, top + 112);
+      core.fillText(
+        "ui",
+        "HTML5魔塔交流群:539113091",
+        text_start,
+        top + 112 + 32
+      );
+      // TODO: 写自己的“关于”页面,每次增加32像素即可
+      core.playSound("打开界面");
+    }
     }
 }
\ No newline at end of file
diff --git a/project/icons.js b/project/icons.js
index f44bfe9..14e3c91 100644
--- a/project/icons.js
+++ b/project/icons.js
@@ -113,7 +113,15 @@ var icons_4665ee12_3a1f_44a4_bea3_0fccba634dc1 =
 		"T392": 81,
 		"T393": 82,
 		"T394": 83,
-		"T395": 84
+		"T395": 84,
+		"T361": 85,
+		"T362": 86,
+		"T363": 87,
+		"T364": 88,
+		"T365": 89,
+		"T397": 90,
+		"T399": 91,
+		"T400": 92
 	},
 	"animates": {
 		"star": 0,
@@ -247,7 +255,11 @@ var icons_4665ee12_3a1f_44a4_bea3_0fccba634dc1 =
 		"blueKing": 45,
 		"keiskeiFairy": 71,
 		"tulipFairy": 72,
-		"purpleBowman": 65
+		"purpleBowman": 65,
+		"E421": 73,
+		"E422": 73,
+		"E425": 74,
+		"E426": 75
 	},
 	"enemy48": {
 		"angel": 0,
@@ -310,7 +322,41 @@ var icons_4665ee12_3a1f_44a4_bea3_0fccba634dc1 =
 		"jumpShoes": 49,
 		"skill1": 30,
 		"wand": 10,
-		"pack": 46
+		"pack": 46,
+		"I366": 5,
+		"I367": 7,
+		"I368": 26,
+		"I369": 31,
+		"I370": 32,
+		"I371": 34,
+		"I372": 35,
+		"I373": 36,
+		"I374": 37,
+		"I375": 38,
+		"I396": 39,
+		"postman": 62,
+		"I401": 63,
+		"I402": 64,
+		"I403": 65,
+		"I404": 66,
+		"I405": 67,
+		"I406": 68,
+		"I407": 69,
+		"I408": 70,
+		"I412": 71,
+		"I409": 72,
+		"I410": 73,
+		"I411": 74,
+		"I413": 75,
+		"I414": 76,
+		"I415": 77,
+		"I416": 78,
+		"I417": 79,
+		"I418": 80,
+		"I419": 81,
+		"I420": 82,
+		"I423": 83,
+		"I424": 84
 	},
 	"autotile": {
 		"autotile": 0,
diff --git a/project/images/005-Attack03.webp b/project/images/005-Attack03.webp
new file mode 100644
index 0000000..09df0ff
Binary files /dev/null and b/project/images/005-Attack03.webp differ
diff --git a/project/images/012-Heal01.webp b/project/images/012-Heal01.webp
new file mode 100644
index 0000000..23c2c6d
Binary files /dev/null and b/project/images/012-Heal01.webp differ
diff --git a/project/images/015-Fire01.webp b/project/images/015-Fire01.webp
new file mode 100644
index 0000000..acd799f
Binary files /dev/null and b/project/images/015-Fire01.webp differ
diff --git a/project/images/jianji.webp b/project/images/jianji.webp
new file mode 100644
index 0000000..40504a4
Binary files /dev/null and b/project/images/jianji.webp differ
diff --git a/project/images/miwu.webp b/project/images/miwu.webp
new file mode 100644
index 0000000..84e72cc
Binary files /dev/null and b/project/images/miwu.webp differ
diff --git a/project/images/sound.webp b/project/images/sound.webp
new file mode 100644
index 0000000..1fed120
Binary files /dev/null and b/project/images/sound.webp differ
diff --git a/project/images/statusBackground.webp b/project/images/statusBackground.webp
new file mode 100644
index 0000000..498f5d8
Binary files /dev/null and b/project/images/statusBackground.webp differ
diff --git a/project/images/tati_310101.webp b/project/images/tati_310101.webp
new file mode 100644
index 0000000..e5a5dce
Binary files /dev/null and b/project/images/tati_310101.webp differ
diff --git a/project/images/winskin.webp b/project/images/winskin.webp
index fb44533..00580fe 100644
Binary files a/project/images/winskin.webp and b/project/images/winskin.webp differ
diff --git a/project/items.js b/project/items.js
index fddc8ac..53adda0 100644
--- a/project/items.js
+++ b/project/items.js
@@ -37,61 +37,23 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"canUseItemEffect": "true"
 	},
 	"greenGem": {
-		"cls": "tools",
+		"cls": "items",
 		"name": "绿宝石",
-		"text": ",护盾+${core.values.greenGem}",
-		"itemEffect": "core.status.hero.mdef += core.values.greenGem * core.status.thisMap.ratio",
-		"itemEffectTip": ",护盾+${core.values.greenGem * core.status.thisMap.ratio}",
-		"useItemEffect": "core.status.hero.mdef += core.values.greenGem",
+		"text": ",法强+${core.values.greenGem}",
+		"itemEffect": "core.status.hero.spell += core.values.greenGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",法强+${core.values.greenGem * core.status.thisMap.ratio}",
+		"useItemEffect": "core.status.hero.spell += core.values.greenGem",
 		"canUseItemEffect": "true"
 	},
 	"yellowGem": {
 		"cls": "items",
 		"name": "黄宝石",
-		"text": "可以进行加点",
-		"itemEffect": "core.status.hero.hp+=1000;core.status.hero.atk+=6;core.status.hero.def+=6;core.status.hero.mdef+=10;",
-		"itemEffectTip": ",全属性提升",
-		"useItemEvent": [
-			{
-				"type": "choices",
-				"choices": [
-					{
-						"text": "攻击+1",
-						"action": [
-							{
-								"type": "setValue",
-								"name": "status:atk",
-								"operator": "+=",
-								"value": "1"
-							}
-						]
-					},
-					{
-						"text": "防御+2",
-						"action": [
-							{
-								"type": "setValue",
-								"name": "status:def",
-								"operator": "+=",
-								"value": "2"
-							}
-						]
-					},
-					{
-						"text": "生命+200",
-						"action": [
-							{
-								"type": "setValue",
-								"name": "status:hp",
-								"operator": "+=",
-								"value": "200"
-							}
-						]
-					}
-				]
-			}
-		],
-		"canUseItemEffect": "true"
+		"text": ",速度+${core.values.yellowGem}",
+		"itemEffect": "core.status.hero.speed += core.values.yellowGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",速度+${core.values.yellowGem * core.status.thisMap.ratio}",
+		"useItemEvent": null,
+		"canUseItemEffect": "true",
+		"useItemEffect": "core.status.hero.spell += core.values.yellowGem"
 	},
 	"redPotion": {
 		"cls": "items",
@@ -130,18 +92,20 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"canUseItemEffect": "true"
 	},
 	"sword0": {
-		"cls": "items",
-		"name": "破旧的剑",
+		"cls": "equips",
+		"name": "基础匕首",
 		"text": "一把已经生锈的剑",
 		"equip": {
 			"type": 0,
 			"animate": "sword",
 			"value": {
-				"atk": 0
-			}
+				"speed": 20
+			},
+			"percentage": {}
 		},
 		"itemEffect": "core.status.hero.atk += 0",
-		"itemEffectTip": ",攻击+0"
+		"itemEffectTip": ",攻击+0",
+		"equipCls": "匕首"
 	},
 	"sword1": {
 		"cls": "equips",
@@ -150,10 +114,10 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"equip": {
 			"type": 0,
 			"animate": "sword",
-			"value": {
-				"atk": 10
-			},
-			"percentage": {}
+			"value": {},
+			"percentage": {
+				"mdef": 10
+			}
 		},
 		"itemEffect": "core.status.hero.atk += 10",
 		"itemEffectTip": ",攻击+10",
@@ -166,10 +130,10 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"equip": {
 			"type": "武器",
 			"animate": "sword",
-			"value": {
-				"atk": 20
-			},
-			"percentage": {}
+			"value": {},
+			"percentage": {
+				"mdef": 20
+			}
 		},
 		"itemEffect": "core.status.hero.atk += 20",
 		"itemEffectTip": ",攻击+20",
@@ -247,10 +211,10 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"text": "一个很普通的铁盾",
 		"equip": {
 			"type": 1,
-			"value": {
-				"def": 10
-			},
-			"percentage": {}
+			"value": {},
+			"percentage": {
+				"mdef": 10
+			}
 		},
 		"itemEffect": "core.status.hero.def += 10",
 		"itemEffectTip": ",防御+10",
@@ -310,13 +274,15 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"itemEffectTip": ",防御+100,护盾+100"
 	},
 	"superPotion": {
-		"cls": "items",
+		"cls": "tools",
 		"name": "圣水",
 		"itemEffect": "core.status.hero.hp *= 2",
 		"itemEffectTip": ",生命值翻倍",
 		"useItemEffect": "core.status.hero.hp *= 2;core.playSound('回血');",
 		"canUseItemEffect": "true",
-		"text": "生命值翻倍"
+		"text": "生命值翻倍",
+		"canBatchUse": "true",
+		"hideInReplay": true
 	},
 	"book": {
 		"cls": "constants",
@@ -332,7 +298,7 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"text": "可以自由往来去过的楼层",
 		"hideInReplay": true,
 		"hideInToolbox": true,
-		"useItemEffect": "//core.ui.drawFly(core.floorIds.indexOf(core.status.floorId));\ncore.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));",
+		"useItemEffect": "//core.ui.drawFly(core.floorIds.indexOf(core.status.floorId));\nflags.canMoveFloor = core.canMoveFloor()\nif (!main.replayChecking && !core.isReplaying()) core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));",
 		"canUseItemEffect": "(function () {\n\tif (core.flags.flyNearStair && !core.nearStair() && !core.canMoveFloor())\n\t\treturn false;\n\treturn core.status.maps[core.status.floorId].canFlyFrom;\n})();"
 	},
 	"coin": {
@@ -546,5 +512,306 @@ var items_296f5d02_12fd_4166_a7c1_b5e830c9ee3a =
 		"name": "钱袋",
 		"itemEffect": "core.status.hero.money += 500",
 		"itemEffectTip": ",金币+500"
+	},
+	"I366": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I367": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I368": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I369": {
+		"cls": "constants",
+		"name": "动画开关",
+		"canUseItemEffect": "true",
+		"hideInToolbox": true,
+		"hideInReplay": true,
+		"useItemEvent": [
+			{
+				"type": "confirm",
+				"text": "是否开启战斗动画",
+				"yes": [
+					{
+						"type": "setValue",
+						"name": "flag:noAnimate",
+						"value": "false"
+					}
+				],
+				"no": [
+					{
+						"type": "setValue",
+						"name": "flag:noAnimate",
+						"value": "true"
+					}
+				]
+			}
+		],
+		"text": "调整战斗动画"
+	},
+	"I370": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I371": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I372": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I373": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I374": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I375": {
+		"cls": "equips",
+		"name": "基础法杖",
+		"canUseItemEffect": "true",
+		"equipCls": "法杖",
+		"equip": {
+			"type": 0,
+			"value": {
+				"matk": 10,
+				"mhp": -100
+			},
+			"percentage": {}
+		}
+	},
+	"I396": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"postman": {
+		"cls": "constants",
+		"name": "言灵",
+		"canUseItemEffect": "true",
+		"text": "可以发送和接收在线留言。",
+		"hideInToolbox": true,
+		"hideInReplay": true,
+		"useItemEvent": [
+			{
+				"type": "while",
+				"condition": "true",
+				"data": [
+					{
+						"type": "choices",
+						"text": "本功能需要您在h5mota.com在线游玩并处于登录状态时使用,可以发送和接受留言。\n发言后需要选择\"获取最新留言\"才能看到自己新发的留言。",
+						"choices": [
+							{
+								"text": "地图上显示在线留言:${flag:comment?'开':'关'}",
+								"icon": "postman",
+								"action": [
+									{
+										"type": "function",
+										"function": "function(){\nif (core.hasFlag('comment')) {\n\tcore.setFlag('comment', false);\n\tcore.plugin.clearCommentSign();\n} else {\n\tcore.setFlag('comment', true);\n\tcore.plugin.drawCommentSign();\n}\n}"
+									}
+								]
+							},
+							{
+								"text": "写留言",
+								"action": [
+									{
+										"type": "input2",
+										"text": "请输入要发送的评论,文明友善发言,拒绝放假、剧透。"
+									},
+									{
+										"type": "function",
+										"function": "function(){\nconst input = core.getFlag('input', '');\nconst tags = [core.status.floorId,\n\tcore.getHeroLoc().x.toString(), core.getHeroLoc().y.toString()\n]\nif (!core.isReplaying()) {\n\tcore.plugin.postComment(input, tags);\n}\n}"
+									},
+									{
+										"type": "function",
+										"function": "function(){\nif (!core.isReplaying()) {\n\tcore.plugin.getComment();\n\tsetTimeout(core.plugin.drawCommentSign, 1000);\n}\n}"
+									}
+								]
+							},
+							{
+								"text": "获取最新留言",
+								"action": [
+									{
+										"type": "function",
+										"function": "function(){\nif (!core.isReplaying()) {\n\tcore.plugin.getComment();\n\tsetTimeout(core.plugin.drawCommentSign, 1000);\n}\n}"
+									}
+								]
+							},
+							{
+								"text": "退出",
+								"action": [
+									{
+										"type": "exit"
+									}
+								]
+							}
+						]
+					}
+				]
+			}
+		]
+	},
+	"I401": {
+		"cls": "items",
+		"name": "初等红宝石",
+		"text": ",攻击+${core.values.redGem}",
+		"itemEffect": "core.status.hero.atk += core.values.redGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",攻击+${core.values.redGem * core.status.thisMap.ratio}",
+		"useItemEffect": "core.status.hero.atk += core.values.redGem",
+		"canUseItemEffect": "true"
+	},
+	"I402": {
+		"cls": "items",
+		"name": "初等蓝宝石",
+		"text": ",防御+${core.values.blueGem}",
+		"itemEffect": "core.status.hero.def += core.values.blueGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",防御+${core.values.blueGem * core.status.thisMap.ratio}",
+		"useItemEffect": "core.status.hero.def += core.values.blueGem",
+		"canUseItemEffect": "true"
+	},
+	"I403": {
+		"cls": "items",
+		"name": "初等绿宝石",
+		"text": ",法强+${core.values.greenGem}",
+		"itemEffect": "core.status.hero.spell += core.values.greenGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",法强+${core.values.greenGem * core.status.thisMap.ratio}",
+		"useItemEffect": "core.status.hero.spell += core.values.greenGem",
+		"canUseItemEffect": "true"
+	},
+	"I404": {
+		"cls": "items",
+		"name": "初等黄宝石",
+		"text": ",速度+${core.values.yellowGem}",
+		"itemEffect": "core.status.hero.speed += core.values.yellowGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",速度+${core.values.yellowGem * core.status.thisMap.ratio}",
+		"useItemEvent": null,
+		"canUseItemEffect": "true",
+		"useItemEffect": "core.status.hero.spell += core.values.yellowGem"
+	},
+	"I405": {
+		"cls": "items",
+		"name": "中等红宝石",
+		"text": ",攻击+${core.values.redGem}",
+		"itemEffect": "core.status.hero.atk += 3 * core.values.redGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",攻击+${core.values.redGem * core.status.thisMap.ratio}",
+		"useItemEffect": "core.status.hero.atk += core.values.redGem",
+		"canUseItemEffect": "true"
+	},
+	"I406": {
+		"cls": "items",
+		"name": "中等蓝宝石",
+		"text": ",防御+${core.values.blueGem}",
+		"itemEffect": "core.status.hero.def += 3 * core.values.blueGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",防御+${core.values.blueGem * core.status.thisMap.ratio}",
+		"useItemEffect": "core.status.hero.def += core.values.blueGem",
+		"canUseItemEffect": "true"
+	},
+	"I407": {
+		"cls": "items",
+		"name": "中等绿宝石",
+		"text": ",法强+${core.values.greenGem}",
+		"itemEffect": "core.status.hero.spell += 3 * core.values.greenGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",法强+${core.values.greenGem * core.status.thisMap.ratio}",
+		"useItemEffect": "core.status.hero.spell += core.values.greenGem",
+		"canUseItemEffect": "true"
+	},
+	"I408": {
+		"cls": "items",
+		"name": "中等黄宝石",
+		"text": ",速度+${core.values.yellowGem}",
+		"itemEffect": "core.status.hero.speed += 3 * core.values.yellowGem * core.status.thisMap.ratio",
+		"itemEffectTip": ",速度+${core.values.yellowGem * core.status.thisMap.ratio}",
+		"useItemEvent": null,
+		"canUseItemEffect": "true",
+		"useItemEffect": "core.status.hero.spell += core.values.yellowGem"
+	},
+	"I412": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I409": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I410": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I411": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I413": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I414": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I415": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I416": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I417": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I418": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I419": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I420": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
+	},
+	"I423": {
+		"cls": "items",
+		"name": "银币",
+		"text": ",生命+${core.values.redPotion}",
+		"itemEffect": "core.status.hero.money += 10",
+		"itemEffectTip": ",金钱+10",
+		"useItemEffect": "core.status.hero.hp += core.values.redPotion",
+		"canUseItemEffect": "true"
+	},
+	"I424": {
+		"cls": "items",
+		"name": "新物品",
+		"canUseItemEffect": "true"
 	}
 }
\ No newline at end of file
diff --git a/project/maps.js b/project/maps.js
index fc722c6..e9b9cf2 100644
--- a/project/maps.js
+++ b/project/maps.js
@@ -1,8 +1,8 @@
 var maps_90f36752_8815_4be8_b32b_d7fad1d0542e = 
 {
-	"1": {"cls":"animates","id":"yellowWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{}}},
-	"2": {"cls":"animates","id":"whiteWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{}}},
-	"3": {"cls":"animates","id":"blueWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{}}},
+	"1": {"cls":"animates","id":"yellowWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{}}},
+	"2": {"cls":"animates","id":"whiteWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{}}},
+	"3": {"cls":"animates","id":"blueWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{}}},
 	"4": {"cls":"animates","id":"star","name":"星空"},
 	"5": {"cls":"animates","id":"lava","name":"岩浆"},
 	"6": {"cls":"animates","id":"ice","doorInfo":{"time":160,"openSound":"破冰镐","closeSound":"door.opus","keys":{"icePickaxe":1}},"animate":1},
@@ -68,12 +68,12 @@ var maps_90f36752_8815_4be8_b32b_d7fad1d0542e =
 	"71": {"cls":"items","id":"shield0"},
 	"72": {"cls":"items","id":"skill1"},
 	"73": {"cls":"items","id":"wand"},
-	"81": {"cls":"animates","id":"yellowDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"yellowKey":1}},"name":"黄门"},
-	"82": {"cls":"animates","id":"blueDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"blueKey":1}},"name":"蓝门"},
-	"83": {"cls":"animates","id":"redDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"redKey":1}},"name":"红门"},
-	"84": {"cls":"animates","id":"greenDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"greenKey":1}},"name":"绿门"},
-	"85": {"cls":"animates","id":"specialDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"specialKey":1}},"name":"机关门"},
-	"86": {"cls":"animates","id":"steelDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"steelKey":1}},"name":"铁门"},
+	"81": {"cls":"animates","id":"yellowDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"yellowKey":1}},"name":"黄门"},
+	"82": {"cls":"animates","id":"blueDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"blueKey":1}},"name":"蓝门"},
+	"83": {"cls":"animates","id":"redDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"redKey":1}},"name":"红门"},
+	"84": {"cls":"animates","id":"greenDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"greenKey":1}},"name":"绿门"},
+	"85": {"cls":"animates","id":"specialDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"specialKey":1}},"name":"机关门"},
+	"86": {"cls":"animates","id":"steelDoor","trigger":"openDoor","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"steelKey":1}},"name":"铁门"},
 	"87": {"cls":"terrains","id":"upFloor","canPass":true},
 	"88": {"cls":"terrains","id":"downFloor","canPass":true},
 	"89": {"cls":"animates","id":"portal","canPass":true},
@@ -86,7 +86,7 @@ var maps_90f36752_8815_4be8_b32b_d7fad1d0542e =
 	"102": {"cls":"animates","id":"crystalBottom"},
 	"103": {"cls":"animates","id":"fire"},
 	"104": {"cls":"animates","id":"switch"},
-	"109": {"cls":"animates","id":"magentaWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{}}},
+	"109": {"cls":"animates","id":"magentaWall","canBreak":true,"animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{}}},
 	"121": {"cls":"npcs","id":"man"},
 	"122": {"cls":"npcs","id":"trader"},
 	"123": {"cls":"npcs","id":"thief"},
@@ -217,12 +217,12 @@ var maps_90f36752_8815_4be8_b32b_d7fad1d0542e =
 	"316": {"cls":"terrains","id":"sWallTLR","name":"薄墙-上左右","cannotOut":["up","left","right"],"cannotIn":["up","left","right"]},
 	"317": {"cls":"terrains","id":"sWallTBR","name":"薄墙-上下右","cannotOut":["up","down","right"],"cannotIn":["up","down","right"]},
 	"318": {"cls":"terrains","id":"sWallTBL","name":"薄墙-上下左","cannotOut":["up","down","left"],"cannotIn":["up","down","left"]},
-	"319": {"cls":"npc48","id":"tallYellowDoor","trigger":"openDoor","name":"高黄门","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"yellowKey":1}}},
-	"320": {"cls":"npc48","id":"tallBlueDoor","trigger":"openDoor","name":"高蓝门","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"blueKey":1}}},
-	"321": {"cls":"npc48","id":"tallRedDoor","trigger":"openDoor","name":"高红门","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"redKey":1}}},
-	"322": {"cls":"npc48","id":"tallGreenDoor","trigger":"openDoor","name":"高绿门","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"greenKey":1}}},
-	"323": {"cls":"npc48","id":"tallSpecialDoor","trigger":"openDoor","name":"高机关门","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"specialKey":1}}},
-	"324": {"cls":"npc48","id":"tallSteelDoor","trigger":"openDoor","name":"高铁门","animate":1,"doorInfo":{"time":160,"openSound":"door.mp3","closeSound":"door.opus","keys":{"steelKey":1}}},
+	"319": {"cls":"npc48","id":"tallYellowDoor","trigger":"openDoor","name":"高黄门","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"yellowKey":1}}},
+	"320": {"cls":"npc48","id":"tallBlueDoor","trigger":"openDoor","name":"高蓝门","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"blueKey":1}}},
+	"321": {"cls":"npc48","id":"tallRedDoor","trigger":"openDoor","name":"高红门","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"redKey":1}}},
+	"322": {"cls":"npc48","id":"tallGreenDoor","trigger":"openDoor","name":"高绿门","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"greenKey":1}}},
+	"323": {"cls":"npc48","id":"tallSpecialDoor","trigger":"openDoor","name":"高机关门","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"specialKey":1}}},
+	"324": {"cls":"npc48","id":"tallSteelDoor","trigger":"openDoor","name":"高铁门","animate":1,"doorInfo":{"time":160,"openSound":"door.opus","closeSound":"door.opus","keys":{"steelKey":1}}},
 	"325": {"cls":"enemys","id":"keiskeiFairy"},
 	"326": {"cls":"enemys","id":"tulipFairy"},
 	"327": {"cls":"enemy48","id":"bearDown"},
@@ -259,6 +259,21 @@ var maps_90f36752_8815_4be8_b32b_d7fad1d0542e =
 	"358": {"cls":"terrains","id":"T358"},
 	"359": {"cls":"terrains","id":"T359"},
 	"360": {"cls":"terrains","id":"T360"},
+	"361": {"cls":"terrains","id":"T361"},
+	"362": {"cls":"terrains","id":"T362"},
+	"363": {"cls":"terrains","id":"T363"},
+	"364": {"cls":"terrains","id":"T364"},
+	"365": {"cls":"terrains","id":"T365"},
+	"366": {"cls":"items","id":"I366"},
+	"367": {"cls":"items","id":"I367"},
+	"368": {"cls":"items","id":"I368"},
+	"369": {"cls":"items","id":"I369"},
+	"370": {"cls":"items","id":"I370"},
+	"371": {"cls":"items","id":"I371"},
+	"372": {"cls":"items","id":"I372"},
+	"373": {"cls":"items","id":"I373"},
+	"374": {"cls":"items","id":"I374"},
+	"375": {"cls":"items","id":"I375"},
 	"376": {"cls":"terrains","id":"T376"},
 	"377": {"cls":"terrains","id":"T377"},
 	"378": {"cls":"terrains","id":"T378"},
@@ -279,6 +294,37 @@ var maps_90f36752_8815_4be8_b32b_d7fad1d0542e =
 	"393": {"cls":"terrains","id":"T393"},
 	"394": {"cls":"terrains","id":"T394"},
 	"395": {"cls":"terrains","id":"T395"},
+	"396": {"cls":"items","id":"I396"},
+	"397": {"cls":"items","id":"postman"},
+	"398": {"cls":"terrains","id":"T397"},
+	"399": {"cls":"terrains","id":"T399"},
+	"400": {"cls":"terrains","id":"T400"},
+	"401": {"cls":"items","id":"I401"},
+	"402": {"cls":"items","id":"I402"},
+	"403": {"cls":"items","id":"I403"},
+	"404": {"cls":"items","id":"I404"},
+	"405": {"cls":"items","id":"I405"},
+	"406": {"cls":"items","id":"I406"},
+	"407": {"cls":"items","id":"I407"},
+	"408": {"cls":"items","id":"I408"},
+	"409": {"cls":"items","id":"I409"},
+	"410": {"cls":"items","id":"I410"},
+	"411": {"cls":"items","id":"I411"},
+	"412": {"cls":"items","id":"I412"},
+	"413": {"cls":"items","id":"I413"},
+	"414": {"cls":"items","id":"I414"},
+	"415": {"cls":"items","id":"I415"},
+	"416": {"cls":"items","id":"I416"},
+	"417": {"cls":"items","id":"I417"},
+	"418": {"cls":"items","id":"I418"},
+	"419": {"cls":"items","id":"I419"},
+	"420": {"cls":"items","id":"I420"},
+	"421": {"cls":"enemys","id":"E421"},
+	"422": {"cls":"enemys","id":"E422"},
+	"423": {"cls":"items","id":"I423"},
+	"424": {"cls":"items","id":"I424"},
+	"425": {"cls":"enemys","id":"E425"},
+	"426": {"cls":"enemys","id":"E426"},
 	"20034": {"cls":"tileset","id":"X20034","canPass":true},
 	"20154": {"cls":"tileset","id":"X20154","canPass":true},
 	"20216": {"cls":"tileset","id":"X20216","canPass":true},
diff --git a/project/materials/enemys.png b/project/materials/enemys.png
index 2b3641c..44fb811 100644
Binary files a/project/materials/enemys.png and b/project/materials/enemys.png differ
diff --git a/project/materials/items.png b/project/materials/items.png
index 04dc29b..bf1a755 100644
Binary files a/project/materials/items.png and b/project/materials/items.png differ
diff --git a/project/materials/terrains.png b/project/materials/terrains.png
index 378c4bf..c68f818 100644
Binary files a/project/materials/terrains.png and b/project/materials/terrains.png differ
diff --git a/project/plugins.js b/project/plugins.js
index 009ae14..cc38544 100644
--- a/project/plugins.js
+++ b/project/plugins.js
@@ -1,226 +1,326 @@
 var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 = 
 {
     "init": function () {
-    this._afterLoadResources = function () {
-      // 本函数将在所有资源加载完毕后,游戏开启前被执行
-      core.ui.statusBar.init();
-      core.registerEvent("changeMouse", function (data) {
-        if (!main.replayChecking && !core.isReplaying())
-          core.changeMouse(
-            data.icon,
-            data.div,
-            data.translate[0],
-            data.translate[1],
-            data.scale[0],
-            data.scale[1],
-            data.angel,
-            data.px,
-            data.py
-          );
-        core.doAction();
-      });
-      core.registerEvent("removeMouse", function (data) {
-        if (!main.replayChecking && !core.isReplaying())
-          core.removeMouse(data.div);
-        core.doAction();
-      });
-      core.registerEvent("addPop", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          data.value = core.replaceText(data.value);
-          core.addPop(
-            data.value,
-            data.px,
-            data.py,
-            data.color,
-            data.boldColor,
-            data.left,
-            data.jump,
-            data.time,
-            data.show,
-            data.font,
-            data.speed
-          );
-        }
-        core.doAction();
-      });
-      core.registerEvent("drawWarning", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          data.text = core.replaceText(data.text);
-          data.text2 = core.replaceText(data.text2);
-          core.drawWarning(
-            data.x,
-            data.y,
-            data.size,
-            data?.text,
-            data?.text2,
-            data?.warning
-          );
-          setTimeout(() => core.doAction(), 3100);
-        } else {
-          core.doAction();
-        }
-      });
-      core.registerEvent("playStereo", function (data) {
-        if (!main.replayChecking && !core.isReplaying())
-          core.playStereo(data.name, data.left, data.right, data.split);
-        core.doAction();
-      });
-      core.registerEvent("moveStereo", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          const id = core.playStereo(
-            data.name,
-            data.left,
-            data.right,
-            data.split
-          );
-          core.moveStereo(id, data.leftTo, data.rightTo, data.time);
-        }
-        core.doAction();
-      });
-      core.registerEvent("over", function (data) {
-        let image = data.image ?? "";
-        let time = data.time ?? 3000;
-        let sound = data.sound ?? "";
-        let textColor = data.textColor ?? "#FFFFFF";
-        let boldColor = data.boldColor ?? "#000000";
-        let font = data.font ?? "bold 48px Verdana";
-        let text = data.text ?? "";
-        let hidetime = data.hidetime ?? 100;
-        if (!main.replayChecking && !core.isReplaying()) {
-          core.over(
-            image,
-            data.memory,
-            time,
-            hidetime,
-            sound,
-            textColor,
-            boldColor,
-            font,
-            text
-          );
-        } else {
-          core.doAction();
-        }
-      });
-      core.registerEvent("changebg", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          core.changebg(
-            data.img1,
-            data.memory1,
-            data.img2,
-            data.memory2,
-            data.time,
-            data.style
-          );
-        } else {
-          core.doAction();
-        }
-      });
-      core.registerEvent("overlist", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          core.overlist(
-            data.image,
-            data.memory,
-            data.hidetime || 30,
-            data.list || [
-              {
-                text: "",
-                sound: "",
-                time: 50,
-                textColor: "#FFFFFF",
-                boldColor: "#000000",
-                font: "bold 48px Verdana",
-                frame: 0,
-              },
-            ]
-          );
-        } else {
-          core.doAction();
-        }
-      });
-      core.registerEvent("op", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          core.openvideo();
-        } else {
-          core.doAction();
-        }
-      });
-      core.registerEvent("animationDrawable", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          core.animationDrawable(
-            data.allFarme,
-            data.color,
-            data.globalAlpha,
-            data.imageList,
-            data.soundList
-          );
-        } else {
-          core.doAction();
-        }
-      });
-      core.registerEvent("setanimate", function (data) {
-        data.px = data.px ?? 0;
-        data.py = data.py ?? 0;
-        core.setanimate(
-          data.name,
-          data.px,
-          data.py,
-          data.width,
-          data.height,
-          data.allFarme,
-          data.imageList,
-          data.soundList
-        );
-        core.doAction();
-      });
-      core.registerEvent("clearanimate", function (data) {
-        core.plugin.playing.clear();
+	this._afterLoadResources = function () {
+		// 本函数将在所有资源加载完毕后,游戏开启前被执行
+		core.ui.statusBar.init();
+		core.dom.playGame.style.fontFamily = "pala";
+		core.dom.loadGame.style.fontFamily = "pala";
+		core.dom.CGMode.style.fontFamily = "pala";
+		core.dom.musicMode.style.fontFamily = "pala";
+		core.dom.replayGame.style.fontFamily = "pala";
+		core.registerEvent("changeMouse", function (data) {
+			if (!main.replayChecking && !core.isReplaying())
+				core.changeMouse(
+					data.icon,
+					data.div,
+					data.translate[0],
+					data.translate[1],
+					data.scale[0],
+					data.scale[1],
+					data.angel,
+					data.px,
+					data.py
+				);
+			core.doAction();
+		});
+		core.registerEvent("removeMouse", function (data) {
+			if (!main.replayChecking && !core.isReplaying())
+				core.removeMouse(data.div);
+			core.doAction();
+		});
+		core.registerEvent("animationDrawable", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.animationDrawable(
+					data.allFarme,
+					data.color,
+					data.globalAlpha,
+					data.imageList,
+					data.soundList
+				);
+			} else {
+				core.doAction();
+			}
+		});
+		core.registerEvent("drawbackground", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.drawbackground(data.image, data.memory);
+				core.ui.cgText.image = data.image;
+				core.ui.cgText.memory = data.memory;
 
-        core.doAction();
-      });
-      core.registerEvent("deleteanimate", function (data) {
-        core.deleteanimate(data.name);
-        core.doAction();
-      });
-      core.registerEvent("playanimate", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          data.x = data.x ?? 0;
-          data.y = data.y ?? 0;
-          data.scalex = data.scalex ?? 1;
-          data.scaley = data.scaley ?? 1;
-          core.playanimate(
-            data.name,
-            data.x,
-            data.y,
-            data.hero,
-            data.scalex,
-            data.scaley
-          );
-          core.doAction();
-        } else {
-          core.doAction();
-        }
-      });
-      core.registerEvent("cgtext", function (data) {
-        if (!main.replayChecking && !core.isReplaying()) {
-          core.ui.cgText.image = data.bg;
-          core.ui.cgText.memory = data.memory;
-          core.ui.cgText.head = core.clone(data.head);
-          core.ui.cgText.name = data.name;
-          core.ui.cgText.text = data.text;
-          core.ui.cgText.time = data.time;
-          core.ui.cgText.wait = data.wait;
-          core.ui.cgText.WindowSkin = data.WindowSkin;
-          core.ui.cgText.sound = data.sound || "";
-          core.ui.cgText.bodyList = core.clone(data.bodyList);
-          main.dom.cgText.style.display = "block";
-          core.ui.cgText.update();
-        } else {
-          core.doAction();
-        }
-      });
-    };
-  },
+			}
+			core.doAction();
+		});
+		core.registerEvent("animateResize", function (data, x, y, prefix) {
+			core.events._action_animateResize(data, x, y, prefix)
+
+		});
+		core.registerEvent("pauseAnimate", function (data, x, y, prefix) {
+			core.maps.pauseAnimate(data.id)
+			core.doAction();
+
+		});
+		core.registerEvent("remuseAnimate", function (data, x, y, prefix) {
+			core.maps.remuseAnimate(data.id)
+			core.doAction();
+		});
+		core.registerEvent("clearbackground", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.clearbackground();
+				core.ui.cgText.image = "";
+			} else {
+
+				core.doAction();
+			}
+		});
+		core.registerEvent("setanimate", function (data) {
+			data.px = data.px ?? 0;
+			data.py = data.py ?? 0;
+			core.setanimate(
+				data.name,
+				data.px,
+				data.py,
+				data.width,
+				data.height,
+				data.allFarme,
+				data.imageList,
+				data.soundList
+			);
+			core.doAction();
+		});
+		core.registerEvent("animatemove", function (data) {
+			core.animatemove(data.id, data.px, data.py, data.relative, data.time, data.style);
+
+			core.doAction();
+		});
+		core.registerEvent("moveAnimate", function (data) {
+			core.maps.moveAnimate(data.id, data.px, data.py, data.relative, data.time, data.style);
+
+			core.doAction();
+		});
+		core.registerEvent("animateloop", function (data) {
+			core.animateloop(data.id, data.loop);
+
+			core.doAction();
+		});
+		core.registerEvent("animatereverse", function (data) {
+			core.animatereverse(data.id, data.reverse);
+
+			core.doAction();
+		});
+		core.registerEvent("animatepause", function (data) {
+			core.animatepause(data.id, data.pause);
+
+			core.doAction();
+		});
+		core.registerEvent("clearanimate", function (data) {
+			core.animateclear(data.id);
+
+			core.doAction();
+		});
+		core.registerEvent("deleteanimate", function (data) {
+			core.deleteanimate(data.name);
+			core.doAction();
+		});
+		core.registerEvent("playanimate", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				data.x = data.x ?? 0;
+				data.y = data.y ?? 0;
+				data.scalex = data.scalex ?? 1;
+				data.scaley = data.scaley ?? 1;
+				core.playanimate(
+					data.name,
+					data.id,
+					data.x,
+					data.y,
+					data.hero,
+					data.scalex,
+					data.scaley,
+					data.loop,
+					data.reverse
+				);
+				core.doAction();
+			} else {
+				core.doAction();
+			}
+		});
+		core.registerEvent("addPop", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				data.value = core.replaceText(data.value);
+				core.addPop(
+					data.value,
+					data.px,
+					data.py,
+					data.color,
+					data.boldColor,
+					data.left,
+					data.jump,
+					data.time,
+					data.show,
+					data.font,
+					data.speed
+				);
+			}
+			core.doAction();
+		});
+		core.registerEvent("drawWarning", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				data.text = core.replaceText(data.text);
+				data.text2 = core.replaceText(data.text2);
+				core.drawWarning(
+					data.x,
+					data.y,
+					data?.text,
+					data?.text2,
+					data?.warning,
+					data.large,
+					data.size
+				);
+
+				setTimeout(() => core.doAction(), 3100);
+			} else {
+				core.doAction();
+			}
+		});
+
+		core.registerEvent("over", function (data) {
+			let image = data.image ?? "";
+			let time = data.time ?? 3000;
+			let sound = data.sound ?? "";
+			let textColor = data.textColor ?? "#FFFFFF";
+			let boldColor = data.boldColor ?? "#000000";
+			let font = data.font ?? "bold 48px Verdana";
+			let text = data.text ?? "";
+			let hidetime = data.hidetime ?? 100;
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.over(
+					image,
+					data.memory,
+					time,
+					hidetime,
+					sound,
+					textColor,
+					boldColor,
+					font,
+					text
+				);
+			} else {
+				core.doAction();
+			}
+		});
+		core.registerEvent("changebg", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.changebg(
+					data.img1,
+					data.memory1,
+					data.img2,
+					data.memory2,
+					data.time,
+					data.style
+				);
+				if (data.style === "引出") { core.ui.cgText.image = ""; } else {} core.ui.cgText.image = data.img2;
+				core.ui.cgText.memory = data.memory2;
+			} else {
+				core.doAction();
+			}
+		});
+		core.registerEvent("overlist", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.overlist(
+					data.image,
+					data.memory,
+					data.hidetime || 30,
+					data.list || [{
+						text: "",
+						sound: "",
+						time: 50,
+						textColor: "#FFFFFF",
+						boldColor: "#000000",
+						font: "bold 48px Verdana",
+						frame: 0,
+					}, ]
+				);
+			} else {
+				core.doAction();
+			}
+		});
+		core.registerEvent("op", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.openvideo();
+			} else {
+				core.doAction();
+			}
+		});
+
+		core.registerEvent("cgtextList", function (data) {
+			core.ui.cgText.textList = data.textList;
+			core.doAction();
+		});
+		core.registerEvent("cgtext", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+
+				core.ui.cgText.head = core.clone(data.head);
+				core.ui.cgText.index = data.index;
+				core.ui.cgText.name = core.plugin[core.ui.cgText.textList][data.index][0];
+				core.ui.cgText.text = data.text ?
+					data.text :
+					core.plugin[core.ui.cgText.textList][data.index][1];
+				core.ui.cgText.time = data.time;
+				core.ui.cgText.wait = data.wait;
+				core.ui.cgText.WindowSkin = data.WindowSkin;
+				core.ui.cgText.sound =
+					data.sound === "" ?
+					data.sound :
+					core.plugin[core.ui.cgText.textList][data.index][2] || "";
+				core.ui.cgText.bodyList = core.clone(data.bodyList);
+				main.dom.cgText.style.display = "block";
+				core.ui.cgText.update();
+			} else {
+				core.doAction();
+			}
+		});
+		core.registerEvent("introAndLoop", function (data) {
+			if (!main.replayChecking && !core.isReplaying()) {
+				core.plugin.introAndLoop(data.intro, data.time, data.loop);
+				core.doAction();
+			} else {
+				core.doAction();
+			}
+		});
+		core.registerEvent("setq", function (data) {
+			core.setFlag("任务地点", data.id);
+
+			core.doAction();
+		});
+
+		core.registerEvent("setmusics", function (data) {
+			if (
+				(core.getLocalStorage("musics") &&
+					core.getLocalStorage("musics").length === 0) ||
+				!core.getLocalStorage("musics")
+			)
+				core.setLocalStorage("musics", ["theme.mp3"]);
+			let a = core.getLocalStorage("musics");
+			if (!data.bgm) {
+				core.setLocalStorage("musics", ["theme.mp3"]);
+			} else {
+				if (!a.includes(data.bgm)) a.push(data.bgm);
+				core.setLocalStorage("musics", a);
+			}
+			core.doAction();
+		});
+		core.registerEvent("setcgs", function (data) {
+			if (!data.img) {
+				core.setLocalStorage("cgs", []);
+			} else {
+				let a = core.getLocalStorage("cgs") ?? [];
+				if (!a.includes(data.img)) a.push(data.img);
+				core.setLocalStorage("cgs", a);
+			}
+			core.doAction();
+		});
+	};
+},
     "drawLight": function () {
     // 绘制灯光/漆黑层效果。调用方式 core.plugin.drawLight(...)
     // 【参数说明】
@@ -264,8 +364,6 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
       ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);
 
       lightDec = core.clamp(lightDec, 0, 1);
-
-      // 绘制每个灯光效果
       ctx.globalCompositeOperation = "destination-out";
       lights.forEach(function (light) {
         // 坐标,半径,中心不透明度
@@ -2484,1560 +2582,2013 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
     })();
   },
     "statusBar": function () {
-    main.dom.floorMsgGroup.style.display = "none";
-    main.dom.statusBar.style.display = "none";
-    main.dom.toolBar.style.display = "none";
-    //所有数据*3是为了实现高清画布
-    const GAMEVIEW_WIDTH = 676 * 3; //横屏画面宽度
-    const GAMEVIEW_HEIGHT = 416 * 3; //横屏画面高度
+	main.dom.floorMsgGroup.style.display = "none";
+	main.dom.statusBar.style.display = "none";
+	main.dom.toolBar.style.display = "none";
+	//所有数据*3是为了实现高清画布
+	const GAMEVIEW_WIDTH = 676 * 3; //横屏画面宽度
+	const GAMEVIEW_HEIGHT = 416 * 3; //横屏画面高度
 
-    const GAMEVIEW_WIDTH_VERTICAL = 416 * 3; //竖屏画面宽度
-    const GAMEVIEW_HEIGHT_VERTICAL = 676 * 3; //竖屏画面高度
+	const GAMEVIEW_WIDTH_VERTICAL = 416 * 3; //竖屏画面宽度
+	const GAMEVIEW_HEIGHT_VERTICAL = 676 * 3; //竖屏画面高度
 
-    const BAR_WIDTH = 130 * 3; //横屏左侧额外距离(即边栏宽度)
-    const BAR_HEIGHT_VERTICAL = 130 * 3; //竖屏上侧额外距离(即边栏高度)
-    const BORDER_WIDTH = 0; //游戏画面左侧偏移距离
-    const BORDER_HEIGHT = 0; //游戏画面上侧偏移距离
+	const BAR_WIDTH = 130 * 3; //横屏左侧额外距离(即边栏宽度)
+	const BAR_HEIGHT_VERTICAL = 130 * 3; //竖屏上侧额外距离(即边栏高度)
+	const BORDER_WIDTH = 0; //游戏画面左侧偏移距离
+	const BORDER_HEIGHT = 0; //游戏画面上侧偏移距离
 
-    const ITEM_BOX_LEFT = 549 * 3; //横屏道具栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
-    const ITEM_BOX_TOP = 155 * 3; //横屏道具栏上侧距离
-    const ITEM_BOX_LEFT_VERTICAL = 160 * 3; //竖屏道具栏左侧距离
-    const ITEM_BOX_TOP_VERTICAL = 549 * 3; //竖屏道具栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
+	const ITEM_BOX_LEFT = 549 * 3; //横屏道具栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
+	const ITEM_BOX_TOP = 175 * 3; //横屏道具栏上侧距离
+	const ITEM_BOX_LEFT_VERTICAL = 160 * 3; //竖屏道具栏左侧距离
+	const ITEM_BOX_TOP_VERTICAL = 549 * 3; //竖屏道具栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
 
-    const EQUIP_BLOCK_LEFT = 549 * 3; //横屏装备栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
-    const EQUIP_BLOCK_TOP = 10 * 3; //横屏装备栏上侧距离
-    const EQUIP_BLOCK_LEFT_VERTICAL = 10 * 3; //竖屏装备栏左侧距离
-    const EQUIP_BLOCK_TOP_VERTICAL = 549 * 3; //竖屏装备栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
+	const EQUIP_BLOCK_LEFT = 549 * 3; //横屏装备栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
+	const EQUIP_BLOCK_TOP = 10 * 3; //横屏装备栏上侧距离
+	const EQUIP_BLOCK_LEFT_VERTICAL = 10 * 3; //竖屏装备栏左侧距离
+	const EQUIP_BLOCK_TOP_VERTICAL = 549 * 3; //竖屏装备栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
 
-    const MAP_BLOCK_LEFT = 551 * 3; //横屏小地图左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
-    const MAP_BLOCK_TOP = 0; //横屏小地图上侧距离
-    const MAP_BLOCK_LEFT_VERTICAL = 0; //竖屏小地图左侧距离
-    const MAP_BLOCK_TOP_VERTICAL = 551 * 3; //竖屏小地图上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
+	const MAP_BLOCK_LEFT = 551 * 3; //横屏小地图左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
+	const MAP_BLOCK_TOP = 0; //横屏小地图上侧距离
+	const MAP_BLOCK_LEFT_VERTICAL = 0; //竖屏小地图左侧距离
+	const MAP_BLOCK_TOP_VERTICAL = 551 * 3; //竖屏小地图上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
 
-    const KEY_BLOCK_LEFT = EQUIP_BLOCK_LEFT; //横屏钥匙栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
-    const KEY_BLOCK_TOP = 110 * 3; //横屏钥匙栏上侧距离
-    const KEY_BLOCK_LEFT_VERTICAL = 110 * 3; //竖屏钥匙栏左侧距离
-    const KEY_BLOCK_TOP_VERTICAL = EQUIP_BLOCK_TOP_VERTICAL; //竖屏钥匙栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
+	const KEY_BLOCK_LEFT = EQUIP_BLOCK_LEFT; //横屏钥匙栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
+	const KEY_BLOCK_TOP = 130 * 3; //横屏钥匙栏上侧距离
+	const KEY_BLOCK_LEFT_VERTICAL = 110 * 3; //竖屏钥匙栏左侧距离
+	const KEY_BLOCK_TOP_VERTICAL = EQUIP_BLOCK_TOP_VERTICAL; //竖屏钥匙栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
 
-    const INFO_BLOCK_LEFT = 10 * 3; //横屏道具说明左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
-    const INFO_BLOCK_TOP = 180 * 3; //横屏道具说明上侧距离
-    const INFO_BLOCK_LEFT_VERTICAL = 113 * 3; //竖屏道具说明左侧距离
-    const INFO_BLOCK_TOP_VERTICAL = 8 * 3; //竖屏道具说明上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
+	const INFO_BLOCK_LEFT = 10 * 3; //横屏道具说明左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
+	const INFO_BLOCK_TOP = 290 * 3; //横屏道具说明上侧距离
+	const INFO_BLOCK_LEFT_VERTICAL = 113 * 3; //竖屏道具说明左侧距离
+	const INFO_BLOCK_TOP_VERTICAL = 8 * 3; //竖屏道具说明上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
 
-    const TOOL_BOX_LEFT = EQUIP_BLOCK_LEFT; //横屏工具栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
-    const TOOL_BOX_TOP = 348 * 3; //横屏工具栏上侧距离
-    const TOOL_BOX_LEFT_VERTICAL = 348 * 3; //竖屏工具栏左侧距离
-    const TOOL_BOX_TOP_VERTICAL = 549 * 3; //竖屏工具栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
+	const TOOL_BOX_LEFT = EQUIP_BLOCK_LEFT; //横屏工具栏左侧距离(右侧边栏需增加BAR_WIDTH+GAMEVIEW_HEIGHT)
+	const TOOL_BOX_TOP = 348 * 3; //横屏工具栏上侧距离
+	const TOOL_BOX_LEFT_VERTICAL = 348 * 3; //竖屏工具栏左侧距离
+	const TOOL_BOX_TOP_VERTICAL = 549 * 3; //竖屏工具栏上侧距离(下侧边栏需增加BAR_HEIGHT_VERTICAL+GAMEVIEW_WIDTH_VERTICAL)
 
-    const TOOL_ICON_OUTER_SIZE = 34 * 3;
+	const TOOL_ICON_OUTER_SIZE = 34 * 3;
 
-    const TEXT_COLOR = "#FFFFFF"; //默认文字颜色
-    const globalAlpha = 0.7; //默认底框透明度
-    const FORCE_COUNTABLE_ITEMS = ["centerFly"]; //常态显示数量的非永久道具,如果道具不在此数组中,则只有道具多余1时显示数量
+	const TEXT_COLOR = "#FFFFFF"; //默认文字颜色
+	const globalAlpha = 0.7; //默认底框透明度
+	const FORCE_COUNTABLE_ITEMS = ["centerFly"]; //常态显示数量的非永久道具,如果道具不在此数组中,则只有道具多余1时显示数量
 
-    const outerBackground = document.createElement("canvas"); //背景画布设置
-    let globalAlphafloor = 0,
-      globalAlphafloorStatus = 4;
-    outerBackground.style.position = "absolute";
-    outerBackground.style.zIndex = 5;
-    outerBackground.id = "outerBackground";
-    main.dom.outerBackground = outerBackground;
-    main.dom.startPanel.insertAdjacentElement("afterend", outerBackground);
+	const outerBackground = document.createElement("canvas"); //背景画布设置
+	let globalAlphafloor = 0,
+		globalAlphafloorStatus = 4;
+	outerBackground.style.position = "absolute";
+	outerBackground.style.zIndex = 5;
+	outerBackground.id = "outerBackground";
+	main.dom.outerBackground = outerBackground;
+	main.dom.startPanel.insertAdjacentElement("afterend", outerBackground);
 
-    const outerUI = document.createElement("canvas"); //额外ui画布设置(状态栏所有绘制、点击都在额外ui上)
-    outerUI.style.position = "absolute";
-    outerUI.style.zIndex = 165;
-    outerUI.id = "outerUI";
+	const outerUI = document.createElement("canvas"); //额外ui画布设置(状态栏所有绘制、点击都在额外ui上)
+	outerUI.style.position = "absolute";
+	outerUI.style.zIndex = 165;
+	outerUI.id = "outerUI";
 
-    main.dom.outerUI = outerUI;
-    outerBackground.insertAdjacentElement("afterend", outerUI);
-    setTimeout(function () {
-      // Should be executed immediately after init()
-      main.canvas.outerUI = outerUI.getContext("2d");
-    });
-    outerUI.onclick = function (e) {
-      try {
-        e.preventDefault();
-        if (!core.isPlaying()) return false;
-        const left = core.dom.gameGroup.offsetLeft;
-        const top = core.dom.gameGroup.offsetTop;
-        const px = Math.floor((e.clientX - left) / core.domStyle.scale),
-          py = Math.floor((e.clientY - top) / core.domStyle.scale);
-        core.ui.statusBar.onclick(px * 3, py * 3);
-      } catch (ee) {
-        main.log(ee);
-      }
-    };
+	main.dom.outerUI = outerUI;
+	outerBackground.insertAdjacentElement("afterend", outerUI);
+	setTimeout(function () {
+		// Should be executed immediately after init()
+		main.canvas.outerUI = outerUI.getContext("2d");
+	});
+	outerUI.onclick = function (e) {
+		try {
+			e.preventDefault();
+			if (!core.isPlaying()) return false;
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+				py = Math.floor((e.clientY - top) / core.domStyle.scale);
+			core.ui.statusBar.onclick(px * 3, py * 3);
+		} catch (ee) {
+			main.log(ee);
+		}
+	};
 
-    const _resize_gameGroup = function (obj) {
-      //游戏画面自适应调节
-      const gameGroup = core.dom.gameGroup;
-      gameGroup.style.width = obj.totalWidth + "px";
-      gameGroup.style.height = obj.totalHeight + "px";
-      gameGroup.style.left = (obj.clientWidth - obj.totalWidth) / 2 + "px";
-      gameGroup.style.top = (obj.clientHeight - obj.totalHeight) / 2 + "px";
-      //floorMsgGroup为切换楼层中生效,显示时间可通过‘全塔属性’——‘切换楼层时间’或游戏内设置调整
-      //显示内容为游戏名/版本号/楼层名
-      // floorMsgGroup
-      var floorMsgGroup = core.dom.floorMsgGroup;
-      var globalAttribute =
-        core.status.globalAttribute || core.initStatus.globalAttribute;
-      floorMsgGroup.style = globalAttribute.floorChangingStyle;
-      floorMsgGroup.style.height = floorMsgGroup.style.width =
-        (GAMEVIEW_HEIGHT / 3) * core.domStyle.scale + "px";
-      floorMsgGroup.style.fontSize = 16 * core.domStyle.scale + "px";
+	const _resize_gameGroup = function (obj) {
+		//游戏画面自适应调节
+		const gameGroup = core.dom.gameGroup;
+		gameGroup.style.width = obj.totalWidth + "px";
+		gameGroup.style.height = obj.totalHeight + "px";
+		gameGroup.style.left = (obj.clientWidth - obj.totalWidth) / 2 + "px";
+		gameGroup.style.top = (obj.clientHeight - obj.totalHeight) / 2 + "px";
+		//floorMsgGroup为切换楼层中生效,显示时间可通过‘全塔属性’——‘切换楼层时间’或游戏内设置调整
+		//显示内容为游戏名/版本号/楼层名
+		// floorMsgGroup
+		var floorMsgGroup = core.dom.floorMsgGroup;
+		var globalAttribute =
+			core.status.globalAttribute || core.initStatus.globalAttribute;
+		floorMsgGroup.style = globalAttribute.floorChangingStyle;
+		floorMsgGroup.style.height = floorMsgGroup.style.width =
+			(GAMEVIEW_HEIGHT / 3) * core.domStyle.scale + "px";
+		floorMsgGroup.style.fontSize = 16 * core.domStyle.scale + "px";
 
-      if (core.domStyle.isVertical) {
-        floorMsgGroup.style.left = "0px";
-        floorMsgGroup.style.top =
-          ((GAMEVIEW_HEIGHT_VERTICAL / 3 - GAMEVIEW_WIDTH_VERTICAL / 3) *
-            core.domStyle.scale) /
-            2 +
-          "px";
-      } else {
-        floorMsgGroup.style.left =
-          ((GAMEVIEW_WIDTH / 3 - GAMEVIEW_HEIGHT / 3) * core.domStyle.scale) /
-            2 +
-          "px";
-        floorMsgGroup.style.top = "0px";
-      }
-      core.dom.musicBtn.style.right =
-        (obj.clientWidth - obj.totalWidth) / 2 + "px";
-      core.dom.musicBtn.style.bottom =
-        (obj.clientHeight - obj.totalHeight) / 2 - 27 + "px";
-      let startBackground = core.domStyle.isVertical
-        ? main.styles.startVerticalBackground || main.styles.startBackground
-        : main.styles.startBackground;
-      if (main.dom.startBackground.getAttribute("__src__") != startBackground) {
-        main.dom.startBackground.setAttribute("__src__", startBackground);
-        main.dom.startBackground.src = startBackground;
-      }
-      const span = document
-        .getElementById("startButtons")
-        .getElementsByTagName("span");
-      let font = (GAMEVIEW_WIDTH / 100) * core.domStyle.scale;
-      if (core.domStyle.isVertical)
-        font = ((GAMEVIEW_WIDTH_VERTICAL * 2) / 100) * core.domStyle.scale;
+		if (core.domStyle.isVertical) {
+			floorMsgGroup.style.left = "0px";
+			floorMsgGroup.style.top =
+				((GAMEVIEW_HEIGHT_VERTICAL / 3 - GAMEVIEW_WIDTH_VERTICAL / 3) *
+					core.domStyle.scale) /
+				2 +
+				"px";
+		} else {
+			floorMsgGroup.style.left =
+				((GAMEVIEW_WIDTH / 3 - GAMEVIEW_HEIGHT / 3) * core.domStyle.scale) /
+				2 +
+				"px";
+			floorMsgGroup.style.top = "0px";
+		}
+		core.dom.musicBtn.style.right =
+			(obj.clientWidth - obj.totalWidth) / 2 + "px";
+		core.dom.musicBtn.style.bottom =
+			(obj.clientHeight - obj.totalHeight) / 2 - 27 + "px";
+		let startBackground = core.domStyle.isVertical ?
+			main.styles.startVerticalBackground || main.styles.startBackground :
+			main.styles.startBackground;
+		if (main.dom.startBackground.getAttribute("__src__") != startBackground) {
+			main.dom.startBackground.setAttribute("__src__", startBackground);
+			main.dom.startBackground.src = startBackground;
+		}
+		const span = document
+			.getElementById("startButtons")
+			.getElementsByTagName("span");
+		let font = (GAMEVIEW_WIDTH / 100) * core.domStyle.scale;
+		if (core.domStyle.isVertical)
+			font = ((GAMEVIEW_WIDTH_VERTICAL * 2) / 100) * core.domStyle.scale;
 
-      core.dom.playGame.style.fontSize = font + "px";
-      core.dom.loadGame.style.fontSize = font + "px";
-      core.dom.CGMode.style.fontSize = font + "px";
-      core.dom.musicMode.style.fontSize = font + "px";
-      core.dom.replayGame.style.fontSize = font + "px";
-      core.dom.startButtonGroup.style.padding = font * 0.3 + "px 25px";
-    };
-    const _resize_canvas = function (obj) {
-      //自适应画布
-      main.dom.outerBackground.style.width = obj.totalWidth + "px";
-      main.dom.outerBackground.style.height = obj.totalHeight + "px";
-      main.dom.outerUI.style.width = obj.totalWidth + "px";
-      main.dom.outerUI.style.height = obj.totalHeight + "px";
-      if (main.dom.CGUI) {
-        main.dom.CGUI.style.width = obj.totalWidth + 3 + "px";
-        main.dom.CGUI.style.height = obj.totalHeight + 3 + "px";
-      }
-      if (main.dom.music) {
-        main.dom.music.style.width = obj.totalWidth + 3 + "px";
-        main.dom.music.style.height = obj.totalHeight + 3 + "px";
-      }
-      if (main.dom.cgText) {
-        main.dom.cgText.style.width = obj.totalWidth + 3 + "px";
-        main.dom.cgText.style.height = obj.totalHeight + 3 + "px";
-      }
-      if (main.dom.over) {
-        main.dom.over.style.width = obj.totalWidth + 3 + "px";
-        main.dom.over.style.height = obj.totalHeight + 3 + "px";
-      }
-      if (main.dom.video) {
-        main.dom.video.style.width = obj.totalWidth + 3 + "px";
-        main.dom.video.style.height = obj.totalHeight + 3 + "px";
-        if (core.domStyle.isVertical)
-          main.dom.video.style.width = obj.totalHeight + 3 + "px";
-        if (core.domStyle.isVertical)
-          main.dom.video.style.height = obj.totalWidth + 3 + "px";
-        main.dom.video.style.top = "50%";
-        main.dom.video.style.left = "50%";
+		core.dom.playGame.style.fontSize = font + "px";
+		core.dom.loadGame.style.fontSize = font + "px";
+		core.dom.CGMode.style.fontSize = font + "px";
+		core.dom.musicMode.style.fontSize = font + "px";
+		core.dom.replayGame.style.fontSize = font + "px";
+		core.dom.startButtonGroup.style.padding = font * 0.3 + "px 25px";
+	};
+	const _resize_canvas = function (obj) {
+		//自适应画布
+		main.dom.outerBackground.style.width = obj.totalWidth + "px";
+		main.dom.outerBackground.style.height = obj.totalHeight + "px";
+		main.dom.outerUI.style.width = obj.totalWidth + "px";
+		main.dom.outerUI.style.height = obj.totalHeight + "px";
+		if (main.dom.CGUI) {
+			main.dom.CGUI.style.width = obj.totalWidth + 3 + "px";
+			main.dom.CGUI.style.height = obj.totalHeight + 3 + "px";
+		}
+		if (main.dom.music) {
+			main.dom.music.style.width = obj.totalWidth + 3 + "px";
+			main.dom.music.style.height = obj.totalHeight + 3 + "px";
+		}
+		if (main.dom.cgText) {
+			main.dom.cgText.style.width = obj.totalWidth + 3 + "px";
+			main.dom.cgText.style.height = obj.totalHeight + 3 + "px";
+		}
+		if (main.dom.logcanvas) {
+			main.dom.logcanvas.style.width = obj.totalWidth + 3 + "px";
+			main.dom.logcanvas.style.height = obj.totalHeight + 3 + "px";
+		}
+		if (main.dom.over) {
+			main.dom.over.style.width = obj.totalWidth + 3 + "px";
+			main.dom.over.style.height = obj.totalHeight + 3 + "px";
+		}
+		if (main.dom.book) {
+			main.dom.book.style.width = obj.totalWidth + 3 + "px";
+			main.dom.book.style.height = obj.totalHeight + 3 + "px";
+		}
+		if (main.dom.video) {
+			main.dom.video.style.width = obj.totalWidth + 3 + "px";
+			main.dom.video.style.height = obj.totalHeight + 3 + "px";
+			if (core.domStyle.isVertical)
+				main.dom.video.style.width = obj.totalHeight + 3 + "px";
+			if (core.domStyle.isVertical)
+				main.dom.video.style.height = obj.totalWidth + 3 + "px";
+			main.dom.video.style.top = "50%";
+			main.dom.video.style.left = "50%";
 
-        main.dom.video.style.transform = "translate(-50%,-50%)";
+			main.dom.video.style.transform = "translate(-50%,-50%)";
 
-        if (core.domStyle.isVertical)
-          main.dom.video.style.transform = "translate(-50%,-50%) rotate(90deg)";
-      }
-      if (main.dom.video1) {
-        main.dom.video1.style.width = obj.totalWidth + 3 + "px";
-        main.dom.video1.style.height = obj.totalHeight + 3 + "px";
-      }
+			if (core.domStyle.isVertical)
+				main.dom.video.style.transform = "translate(-50%,-50%) rotate(90deg)";
+		}
+		if (main.dom.video1) {
+			main.dom.video1.style.width = obj.totalWidth + 3 + "px";
+			main.dom.video1.style.height = obj.totalHeight + 3 + "px";
+		}
 
-      const innerSize = obj.canvasWidth * core.domStyle.scale + "px";
-      for (let i = 0; i < core.dom.gameCanvas.length; ++i)
-        core.dom.gameCanvas[i].style.width = core.dom.gameCanvas[
-          i
-        ].style.height = innerSize;
-      core.dom.gif.style.width = core.dom.gif.style.height = innerSize;
-      core.dom.gif2.style.width = core.dom.gif2.style.height = innerSize;
+		main.dom.boss1.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss1.style.height = obj.totalHeight + 3 + "px";
 
-      core.dom.gameDraw.style.width = core.dom.gameDraw.style.height =
-        innerSize;
-      core.dom.gameDraw.style.top =
-        obj.gameDrawBox.top * core.domStyle.scale + "px";
-      core.dom.gameDraw.style.left =
-        obj.gameDrawBox.left * core.domStyle.scale + "px";
-      // resize bigmap
-      core.bigmap.canvas.forEach(function (cn) {
-        const ratio = core.canvas[cn].canvas.hasAttribute("isHD")
-          ? core.domStyle.ratio
-          : 1;
-        core.canvas[cn].canvas.style.width =
-          (innerSize / ratio) * core.domStyle.scale + "px";
-        core.canvas[cn].canvas.style.height =
-          (innerSize / ratio) * core.domStyle.scale + "px";
-      });
-      // resize dynamic canvas
-      for (const name in core.dymCanvas) {
-        const ctx = core.dymCanvas[name],
-          canvas = ctx.canvas;
-        const ratio = canvas.hasAttribute("isHD") ? core.domStyle.ratio : 1;
-        canvas.style.width = (innerSize / ratio) * core.domStyle.scale + "px";
-        canvas.style.height = (innerSize / ratio) * core.domStyle.scale + "px";
-        canvas.style.left =
-          parseFloat(canvas.getAttribute("_left")) * core.domStyle.scale + "px";
-        canvas.style.top =
-          parseFloat(canvas.getAttribute("_top")) * core.domStyle.scale + "px";
-      }
-      // resize next
-      main.dom.next.style.width = main.dom.next.style.height =
-        5 * core.domStyle.scale + "px";
-      main.dom.next.style.borderBottomWidth =
-        main.dom.next.style.borderRightWidth = 4 * core.domStyle.scale + "px";
-    };
-    const bgctx = main.dom.outerBackground.getContext("2d");
-    const uictx = main.dom.outerUI.getContext("2d");
-    let now = 0;
-    core.registerAnimationFrame("lightFloor", true, function (timestamp) {
-      if (timestamp - now > 1000 / 60) {
-        now = timestamp;
-        globalAlphafloor += globalAlphafloorStatus;
-        if (globalAlphafloor === 100) globalAlphafloorStatus = -2;
-        if (globalAlphafloor === 0) globalAlphafloorStatus = 2;
+		main.dom.boss2.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss2.style.height = obj.totalHeight + 3 + "px";
 
-        if (core.domStyle.isVertical) {
-          core.clearMap(
-            uictx,
-            MAP_BLOCK_LEFT_VERTICAL,
-            MAP_BLOCK_TOP_VERTICAL,
-            340,
-            360
-          );
-          if (core.status.event.id === "viewMaps") {
-            core.ui.statusBar._update_map(core.status.event.data.floorId);
-          } else {
-            core.ui.statusBar._update_map();
-          }
+		main.dom.boss3.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss3.style.height = obj.totalHeight + 3 + "px";
 
-          uictx.globalAlpha = globalAlphafloor / 100;
-          core.drawImage(
-            uictx,
-            "green.webp",
-            MAP_BLOCK_LEFT_VERTICAL + 135,
-            MAP_BLOCK_TOP_VERTICAL + 170
-          );
-          uictx.globalAlpha = 1;
-        } else {
-          core.clearMap(uictx, MAP_BLOCK_LEFT, MAP_BLOCK_TOP, 340, 360);
-          if (core.status.event.id === "viewMaps") {
-            core.ui.statusBar._update_map(core.status.event.data.floorId);
-          } else {
-            core.ui.statusBar._update_map();
-          }
-          uictx.globalAlpha = globalAlphafloor / 100;
-          core.drawImage(
-            uictx,
-            "green.webp",
-            MAP_BLOCK_LEFT + 150,
-            MAP_BLOCK_TOP + 180
-          );
-          uictx.globalAlpha = 1;
-        }
-      }
-    });
+		main.dom.boss4.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss4.style.height = obj.totalHeight + 3 + "px";
 
-    core.control.resize = function () {
-      //自适应,可实现横竖屏切换
-      if (main.mode == "editor") return;
+		main.dom.boss5.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss5.style.height = obj.totalHeight + 3 + "px";
+		main.dom.boss6.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss6.style.height = obj.totalHeight + 3 + "px";
+		main.dom.boss7.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss7.style.height = obj.totalHeight + 3 + "px";
+		main.dom.boss8.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss8.style.height = obj.totalHeight + 3 + "px";
+		main.dom.saveLoad.style.width = obj.totalWidth + 3 + "px";
+		main.dom.saveLoad.style.height = obj.totalHeight + 3 + "px";
+		main.dom.boss.style.width = obj.totalWidth + 3 + "px";
+		main.dom.boss.style.height = obj.totalHeight + 3 + "px";
 
-      const clientWidth = main.dom.body.clientWidth,
-        clientHeight = main.dom.body.clientHeight;
-      const canvasWidth = core.__PIXELS__;
+		const innerSize = obj.canvasWidth * core.domStyle.scale + "px";
+		for (let i = 0; i < core.dom.gameCanvas.length; ++i)
+			core.dom.gameCanvas[i].style.width = core.dom.gameCanvas[
+				i
+			].style.height = innerSize;
+		core.dom.gif.style.width = core.dom.gif.style.height = innerSize;
+		core.dom.gif2.style.width = core.dom.gif2.style.height = innerSize;
 
-      const isVertical = clientHeight > clientWidth;
-      core.domStyle.isVertical = isVertical;
+		core.dom.gameDraw.style.width = core.dom.gameDraw.style.height =
+			innerSize;
+		core.dom.gameDraw.style.top =
+			obj.gameDrawBox.top * core.domStyle.scale + "px";
+		core.dom.gameDraw.style.left =
+			obj.gameDrawBox.left * core.domStyle.scale + "px";
+		// resize bigmap
+		core.bigmap.canvas.forEach(function (cn) {
+			const ratio = core.canvas[cn].canvas.hasAttribute("isHD") ?
+				core.domStyle.ratio :
+				1;
+			core.canvas[cn].canvas.style.width =
+				(innerSize / ratio) * core.domStyle.scale + "px";
+			core.canvas[cn].canvas.style.height =
+				(innerSize / ratio) * core.domStyle.scale + "px";
+		});
+		// resize dynamic canvas
+		for (const name in core.dymCanvas) {
+			const ctx = core.dymCanvas[name],
+				canvas = ctx.canvas;
+			const ratio = canvas.hasAttribute("isHD") ? core.domStyle.ratio : 1;
+			canvas.style.width = (innerSize / ratio) * core.domStyle.scale + "px";
+			canvas.style.height = (innerSize / ratio) * core.domStyle.scale + "px";
+			canvas.style.left =
+				parseFloat(canvas.getAttribute("_left")) * core.domStyle.scale + "px";
+			canvas.style.top =
+				parseFloat(canvas.getAttribute("_top")) * core.domStyle.scale + "px";
+		}
+		// resize next
+		main.dom.next.style.width = main.dom.next.style.height =
+			5 * core.domStyle.scale + "px";
+		main.dom.next.style.borderBottomWidth =
+			main.dom.next.style.borderRightWidth = 4 * core.domStyle.scale + "px";
+	};
+	const bgctx = main.dom.outerBackground.getContext("2d");
+	const uictx = main.dom.outerUI.getContext("2d");
+	let now = 0;
+	core.registerAnimationFrame("lightFloor", true, function (timestamp) {
+		if (!core.status.floorId) return
+		if (timestamp - now > 1000 / 60) {
+			now = timestamp;
+			globalAlphafloor += globalAlphafloorStatus;
+			if (globalAlphafloor === 100) globalAlphafloorStatus = -2;
+			if (globalAlphafloor === 0) globalAlphafloorStatus = 2;
 
-      const totalWidth = isVertical
-          ? GAMEVIEW_WIDTH_VERTICAL / 3
-          : GAMEVIEW_WIDTH / 3,
-        totalHeight = isVertical
-          ? GAMEVIEW_HEIGHT_VERTICAL / 3
-          : GAMEVIEW_HEIGHT / 3;
+			if (core.domStyle.isVertical) {
+				core.clearMap(
+					uictx,
+					MAP_BLOCK_LEFT_VERTICAL,
+					MAP_BLOCK_TOP_VERTICAL,
+					340,
+					360
+				);
+				if (core.status.event.id === "viewMaps") {
+					core.ui.statusBar._update_map(core.status.event.data.floorId);
+				} else {
+					core.ui.statusBar._update_map();
+				}
 
-      const maxRatio = Math.min(
-        clientWidth / totalWidth,
-        clientHeight / totalHeight
-      );
+				uictx.globalAlpha = globalAlphafloor / 100;
+				core.drawImage(
+					uictx,
+					"green.webp",
+					MAP_BLOCK_LEFT_VERTICAL + 135,
+					MAP_BLOCK_TOP_VERTICAL + 170
+				);
+				uictx.globalAlpha = 1;
+			} else {
+				core.clearMap(uictx, MAP_BLOCK_LEFT, MAP_BLOCK_TOP, 340, 360);
+				if (core.status.event.id === "viewMaps") {
+					core.ui.statusBar._update_map(core.status.event.data.floorId);
+				} else {
+					core.ui.statusBar._update_map();
+				}
+				uictx.globalAlpha = globalAlphafloor / 100;
+				core.drawImage(
+					uictx,
+					"green.webp",
+					MAP_BLOCK_LEFT + 150,
+					MAP_BLOCK_TOP + 180
+				);
+				uictx.globalAlpha = 1;
+			}
+		}
+	});
 
-      core.domStyle.availableScale = [];
-      [1, 1.25, 1.5, 1.75, 2].forEach(function (v) {
-        if (maxRatio >= v) {
-          core.domStyle.availableScale.push(v);
-        }
-      });
+	core.control.resize = function () {
+		//自适应,可实现横竖屏切换
+		if (main.mode == "editor") return;
 
-      if (core.domStyle.availableScale.indexOf(core.domStyle.scale) < 0) {
-        core.domStyle.scale = Math.min(1, maxRatio);
-      } else if (
-        core.getLocalStorage("scale") == null &&
-        core.domStyle.availableScale.length >= 2
-      ) {
-        core.domStyle.scale =
-          core.domStyle.availableScale[core.domStyle.availableScale.length - 2];
-        core.setLocalStorage("scale", core.domStyle.scale);
-      }
+		const clientWidth = main.dom.body.clientWidth,
+			clientHeight = main.dom.body.clientHeight;
+		const canvasWidth = core.__PIXELS__;
 
-      const totalWidthScaled = totalWidth * core.domStyle.scale,
-        totalHeightScaled = totalHeight * core.domStyle.scale;
+		const isVertical = clientHeight > clientWidth;
+		core.domStyle.isVertical = isVertical;
 
-      const gameDrawBox = isVertical
-        ? {
-            left: BORDER_WIDTH / 3,
-            top: BAR_HEIGHT_VERTICAL / 3 + BORDER_HEIGHT / 3,
-          }
-        : { left: BAR_WIDTH / 3 + BORDER_WIDTH / 3, top: BORDER_HEIGHT / 3 };
+		const totalWidth = isVertical ?
+			GAMEVIEW_WIDTH_VERTICAL / 3 :
+			GAMEVIEW_WIDTH / 3,
+			totalHeight = isVertical ?
+			GAMEVIEW_HEIGHT_VERTICAL / 3 :
+			GAMEVIEW_HEIGHT / 3;
 
-      const obj = {
-        clientWidth: clientWidth,
-        clientHeight: clientHeight,
-        canvasWidth: canvasWidth,
-        totalWidth: totalWidthScaled,
-        totalHeight: totalHeightScaled,
-        gameDrawBox: gameDrawBox,
-        globalAttribute:
-          core.status.globalAttribute || core.initStatus.globalAttribute,
-      };
+		const maxRatio = Math.min(
+			clientWidth / totalWidth,
+			clientHeight / totalHeight
+		);
 
-      _resize_gameGroup(obj);
-      _resize_canvas(obj);
+		core.domStyle.availableScale = [];
+		[1, 1.25, 1.5, 1.75, 2].forEach(function (v) {
+			if (maxRatio >= v) {
+				core.domStyle.availableScale.push(v);
+			}
+		});
 
-      if (core.status.automaticRoute == null) core.status.automaticRoute = {};
-      core.updateStatusBar();
-      if (main.dom.CGUI && main.dom.CGUI.style.display === "block")
-        core.ui.CG.update();
-      if (main.dom.music && main.dom.music.style.display === "block")
-        core.ui.music.update();
-      if (main.dom.cgText && main.dom.cgText.style.display === "block")
-        core.ui.cgText.update();
-    };
+		if (core.domStyle.availableScale.indexOf(core.domStyle.scale) < 0) {
+			core.domStyle.scale = Math.min(1, maxRatio);
+		} else if (
+			core.getLocalStorage("scale") == null &&
+			core.domStyle.availableScale.length >= 2
+		) {
+			core.domStyle.scale =
+				core.domStyle.availableScale[core.domStyle.availableScale.length - 2];
+			core.setLocalStorage("scale", core.domStyle.scale);
+		}
 
-    class StatusBar {
-      constructor() {
-        //道具栏列表
-        this.itemMx = [
-          //空位用‘none’填充,当前ui至多4列6行
-          ["book", "wand", "none", "fly"],
-          ["cross", "superPotion", "pickaxe"],
-          ["bomb", "centerFly", "upFly"],
-          ["none", "none", "none"],
-          ["downFly", "knife", "snow"],
-          ["bigKey", "earthquake", "coin"],
-        ];
-      }
-      //初始化内容(工具栏/录像操作执行函数)
-      init() {
-        this.toolbarAction = [
-          [
-            main.core.openKeyBoard,
-            main.core.openQuickShop,
-            core.openToolbox,
-            core.doSL,
-          ],
-          [main.core.openSettings, main.core.save, main.core.load, core.doSL],
-        ];
-        this.replayAction = [
-          [core.triggerReplay, core.stopReplay, core.rewindReplay],
-          [core.speedDownReplay, core.speedUpReplay, core.saveReplay],
-        ];
-      }
-      //更新
-      update() {
-        this._update_background(); //更新背景
-        this._update_props(); //更新属性
-        //this._update_items(); //更新道具
-        //this._update_equips(); //更新装备
-        //this._update_keys(); //更新钥匙
-        //this._update_infoWindow(); //更新道具说明
-        this._update_toolBox(); //更新工具栏
-        this._redrawMap();
-      }
-      _redrawMap() {
-        if (core.domStyle.isVertical) {
-          core.clearMap(
-            uictx,
-            MAP_BLOCK_LEFT_VERTICAL,
-            MAP_BLOCK_TOP_VERTICAL,
-            340,
-            360
-          );
-          this._update_map();
-          uictx.globalAlpha = globalAlphafloor / 100;
-          core.drawImage(
-            uictx,
-            "green.webp",
-            MAP_BLOCK_LEFT_VERTICAL + 125,
-            MAP_BLOCK_TOP_VERTICAL + 170
-          );
-          uictx.globalAlpha = 1;
-        } else {
-          core.clearMap(uictx, MAP_BLOCK_LEFT, MAP_BLOCK_TOP, 340, 360);
-          this._update_map();
-          uictx.globalAlpha = globalAlphafloor / 100;
-          core.drawImage(
-            uictx,
-            "green.webp",
-            MAP_BLOCK_LEFT + 150,
-            MAP_BLOCK_TOP + 170
-          );
-          uictx.globalAlpha = 1;
-        }
-      }
-      //更新背景
-      _update_background() {
-        if (core.domStyle.isVertical) {
-          bgctx.canvas.width = GAMEVIEW_WIDTH_VERTICAL;
-          bgctx.canvas.height = GAMEVIEW_HEIGHT_VERTICAL;
-          uictx.canvas.width = GAMEVIEW_WIDTH_VERTICAL;
-          uictx.canvas.height = GAMEVIEW_HEIGHT_VERTICAL;
+		const totalWidthScaled = totalWidth * core.domStyle.scale,
+			totalHeightScaled = totalHeight * core.domStyle.scale;
 
-          const bg = core.material.images.images["status.webp"]; //竖屏背景(上)
-          bgctx.drawImage(
-            bg,
-            0,
-            0,
-            GAMEVIEW_WIDTH_VERTICAL,
-            BAR_HEIGHT_VERTICAL
-          );
-          const bg2 = core.material.images.images["status.webp"]; //竖屏背景(下)
-          bgctx.drawImage(
-            bg2,
-            0,
-            BAR_HEIGHT_VERTICAL + GAMEVIEW_WIDTH_VERTICAL,
-            GAMEVIEW_WIDTH_VERTICAL,
-            BAR_HEIGHT_VERTICAL
-          );
-          bgctx.globalAlpha = globalAlpha;
-          bgctx.globalAlpha = 1;
-          core.setTextAlign("outerUI", "center");
-        } else {
-          bgctx.canvas.width = GAMEVIEW_WIDTH;
-          bgctx.canvas.height = GAMEVIEW_HEIGHT;
-          uictx.canvas.width = GAMEVIEW_WIDTH;
-          uictx.canvas.height = GAMEVIEW_HEIGHT;
+		const gameDrawBox = isVertical ? {
+			left: BORDER_WIDTH / 3,
+			top: BAR_HEIGHT_VERTICAL / 3 + BORDER_HEIGHT / 3,
+		} : { left: BAR_WIDTH / 3 + BORDER_WIDTH / 3, top: BORDER_HEIGHT / 3 };
 
-          const bg = core.material.images.images["status.webp"]; //横屏背景(左)
-          bgctx.drawImage(bg, 0, 0, BAR_WIDTH, GAMEVIEW_HEIGHT);
-          const bg2 = core.material.images.images["status.webp"]; //横屏背景(右)
-          bgctx.drawImage(
-            bg2,
-            BAR_WIDTH + GAMEVIEW_HEIGHT,
-            0,
-            BAR_WIDTH,
-            GAMEVIEW_HEIGHT
-          );
-          bgctx.globalAlpha = globalAlpha;
+		const obj = {
+			clientWidth: clientWidth,
+			clientHeight: clientHeight,
+			canvasWidth: canvasWidth,
+			totalWidth: totalWidthScaled,
+			totalHeight: totalHeightScaled,
+			gameDrawBox: gameDrawBox,
+			globalAttribute: core.status.globalAttribute || core.initStatus.globalAttribute,
+		};
 
-          bgctx.globalAlpha = 1;
-          core.setTextAlign("outerUI", "center");
-        }
-      }
-      // 更新属性
-      _update_props(updatedFloorTitle) {
-        if (!updatedFloorTitle && core.status.floorId) {
-          updatedFloorTitle = core.status.maps[core.status.floorId].title;
-        }
-        const statusList = ["hp", "atk", "def", "money"]; //属性列表,图标在函数复写core.statusBar.icons中声明,数字为project\materials\icons.png中的图标序号(可使用便捷ps追加,第一个序号为0)
-        const drawStatusList = (baseX, baseY) => {
-          let curh = baseY;
-          core.setTextAlign("outerUI", "right");
-          statusList.forEach((item) => {
-            // 绘制图标
-            core.drawIcon(
-              "outerUI",
-              item,
-              baseX - 95 * 3,
-              curh - 18 * 3,
-              22 * 3,
-              22 * 3
-            );
+		_resize_gameGroup(obj);
+		_resize_canvas(obj);
 
-            // 四舍五入
-            core.status.hero[item] = Math.round(core.status.hero[item]);
-            // 大数据格式化
-            core.fillBoldText1(
-              "outerUI",
-              core.getRealStatus(item),
-              baseX,
-              curh,
-              TEXT_COLOR,
-              "#000000",
-              6
-            );
-            curh += 24 * 3;
-            if (curh > 130 * 3 && core.domStyle.isVertical) {
-              curh = 24 * 3;
-              baseX += 105 * 3;
-            }
-          });
-          core.setTextAlign("outerUI", "center");
-        };
-        if (core.domStyle.isVertical) {
-          core.clearMap("outerUI", 10 * 3, 0, 210 * 3, 120 * 3);
-          core.setFont("outerUI", "bold 42px Verdana");
-          if (updatedFloorTitle) {
-            core.fillBoldText1(
-              "outerUI",
-              updatedFloorTitle,
-              60 * 3,
-              22 * 3,
-              TEXT_COLOR,
-              "#000000",
-              6
-            );
-          }
-          //drawStatusList(96 * 3, 46 * 3);
-          //core.drawImage("outerUI", "lane1.png", 0, 0)
-          core.drawImage("outerUI", "cao.webp", 0, 0);
-        } else {
-          core.clearMap("outerUI", 10 * 3, 40 * 3, 105 * 3, 250 * 3);
-          core.setFont("outerUI", "bold 48px Verdana");
-          if (updatedFloorTitle) {
-            core.fillBoldText1(
-              "outerUI",
-              updatedFloorTitle,
-              62 * 3,
-              41 * 3,
-              TEXT_COLOR,
-              "#000000",
-              6
-            );
-          }
-          //drawStatusList(110 * 3, 93 * 3);
-          //core.drawImage("outerUI", "lane1.png", 0, 30)
-          core.drawImage(
-            "outerUI",
-            "cao.webp",
-            0,
-            0,
-            400,
-            350,
-            0,
-            30,
-            360,
-            315
-          );
-        }
-      }
-      _update_items() {
-        //更新道具栏
-        const drawItemMx = (drawFn) => {
-          for (let i = 0; i < this.itemMx.length; i++) {
-            for (let j = 0; j < this.itemMx[i].length; j++) {
-              var item = this.itemMx[i][j];
-              drawFn(i, j, item);
-            }
-          }
-        };
-        const drawItem = (item, posx, posy) => {
-          const icon = core.material.icons.items[item],
-            image = core.material.images.items;
-          core.drawImage(
-            "outerUI",
-            image,
-            0,
-            32 * icon,
-            32,
-            32,
-            posx,
-            posy,
-            30 * 3,
-            30 * 3
-          );
-          const cnt = core.itemCount(item);
-          if (
-            (core.items.items[item].cls === "tools" && cnt > 1) ||
-            FORCE_COUNTABLE_ITEMS.includes(item)
-          ) {
-            core.fillText(
-              "outerUI",
-              cnt,
-              posx + 25 * 3,
-              posy + 28 * 3,
-              "#FFFFFF",
-              "bold 36px Verdana"
-            );
-          }
-        };
-        if (core.domStyle.isVertical) {
-          core.clearMap(
-            "outerUI",
-            ITEM_BOX_LEFT_VERTICAL,
-            ITEM_BOX_TOP_VERTICAL,
-            185 * 3,
-            125 * 3
-          );
+		if (core.status.automaticRoute == null) core.status.automaticRoute = {};
+		core.updateStatusBar();
+		if (main.dom.CGUI && main.dom.CGUI.style.display === "block")
+			core.ui.CG.update();
+		if (main.dom.music && main.dom.music.style.display === "block")
+			core.ui.music.update();
+		if (main.dom.cgText && main.dom.cgText.style.display === "block")
+			core.ui.cgText.update();
+		if (main.dom.logcanvas && main.dom.logcanvas.style.display === "block")
+			core.ui.cgText.update();
+		if (main.dom.boss1 && main.dom.boss1.style.display === "block")
+			core.ui.boss.update();
+		if (main.dom.over && main.dom.over.style.display === "block")
+			core.overupdate()
+		if (main.dom.saveLoad && main.dom.saveLoad.style.display === "block")
+			core.saveLoad.update();
+	};
 
-          drawItemMx((i, j, item) => {
-            if (core.hasItem(item)) {
-              const posx = ITEM_BOX_LEFT_VERTICAL + i * 30 * 3,
-                posy = ITEM_BOX_TOP_VERTICAL + j * 31 * 3;
-              drawItem(item, posx, posy);
-            }
-          });
-        } else {
-          core.clearMap(
-            "outerUI",
-            ITEM_BOX_LEFT,
-            ITEM_BOX_TOP,
-            125 * 3,
-            185 * 3
-          );
+	class StatusBar {
+		constructor() {
+			//道具栏列表
+			this.itemMx = [
+				//空位用‘none’填充,当前ui至多4列6行
+				["book", "postman", "I369", "fly"],
+				["cross", "superPotion", "pickaxe"],
+				["bomb", "centerFly", "upFly"],
+				["none", "none", "none"],
+				["downFly", "knife", "snow"],
+				["bigKey", "earthquake", "coin"],
+			];
+		}
+		//初始化内容(工具栏/录像操作执行函数)
+		init() {
+			this.toolbarAction = [
+				[
+					main.core.openKeyBoard,
+					main.core.openQuickShop,
+					core.openToolbox,
+					core.doSL,
+				],
+				[main.core.save, main.core.load, main.core.openSettings, core.doSL],
+			];
+			this.replayAction = [
+				[core.triggerReplay, core.stopReplay, core.rewindReplay],
+				[
+					core.speedDownReplay,
+					core.speedUpReplay,
+					function () {
+						core.control._replay_SL();
+					},
+				],
+			];
+		}
+		//更新
+		update() {
+			this._update_background(); //更新背景
+			this._update_props(); //更新属性
+			this._update_items(); //更新道具
+			//this._update_equips(); //更新装备
+			this._update_keys(); //更新钥匙
+			//this._update_infoWindow(); //更新道具说明
+			this._update_toolBox(); //更新工具栏
+			this._redrawMap();
+		}
+		_redrawMap() {
+			if (core.domStyle.isVertical) {
+				core.clearMap(
+					uictx,
+					MAP_BLOCK_LEFT_VERTICAL,
+					MAP_BLOCK_TOP_VERTICAL,
+					340,
+					360
+				);
+				this._update_map();
+				uictx.globalAlpha = globalAlphafloor / 100;
+				core.drawImage(
+					uictx,
+					"green.webp",
+					MAP_BLOCK_LEFT_VERTICAL + 125,
+					MAP_BLOCK_TOP_VERTICAL + 170
+				);
+				uictx.globalAlpha = 1;
+			} else {
+				core.clearMap(uictx, MAP_BLOCK_LEFT, MAP_BLOCK_TOP, 340, 360);
+				this._update_map();
+				uictx.globalAlpha = globalAlphafloor / 100;
+				core.drawImage(
+					uictx,
+					"green.webp",
+					MAP_BLOCK_LEFT + 150,
+					MAP_BLOCK_TOP + 170
+				);
+				uictx.globalAlpha = 1;
+			}
+		}
+		//更新背景
+		_update_background() {
+			if (core.domStyle.isVertical) {
+				bgctx.canvas.width = GAMEVIEW_WIDTH_VERTICAL;
+				bgctx.canvas.height = GAMEVIEW_HEIGHT_VERTICAL;
+				uictx.canvas.width = GAMEVIEW_WIDTH_VERTICAL;
+				uictx.canvas.height = GAMEVIEW_HEIGHT_VERTICAL;
 
-          drawItemMx((i, j, item) => {
-            if (core.hasItem(item)) {
-              const posx = ITEM_BOX_LEFT + j * 30 * 3,
-                posy = ITEM_BOX_TOP + i * 31 * 3;
-              drawItem(item, posx, posy);
-            }
-          });
-        }
-      }
+				const bg = core.material.images.images["status.webp"]; //竖屏背景(上)
+				bgctx.drawImage(
+					bg,
+					0,
+					0,
+					GAMEVIEW_WIDTH_VERTICAL,
+					BAR_HEIGHT_VERTICAL
+				);
+				const bg2 = core.material.images.images["status.webp"]; //竖屏背景(下)
+				bgctx.drawImage(
+					bg2,
+					0,
+					BAR_HEIGHT_VERTICAL + GAMEVIEW_WIDTH_VERTICAL,
+					GAMEVIEW_WIDTH_VERTICAL,
+					BAR_HEIGHT_VERTICAL
+				);
+				bgctx.globalAlpha = globalAlpha;
+				bgctx.globalAlpha = 1;
+				core.setTextAlign("outerUI", "center");
+			} else {
+				bgctx.canvas.width = GAMEVIEW_WIDTH;
+				bgctx.canvas.height = GAMEVIEW_HEIGHT;
+				uictx.canvas.width = GAMEVIEW_WIDTH;
+				uictx.canvas.height = GAMEVIEW_HEIGHT;
 
-      _update_map(floorId = core.status.floorId) {
-        const x = core.domStyle.isVertical
-          ? MAP_BLOCK_LEFT_VERTICAL
-          : MAP_BLOCK_LEFT;
-        const y = core.domStyle.isVertical
-          ? MAP_BLOCK_TOP_VERTICAL
-          : MAP_BLOCK_TOP;
+				const bg = core.material.images.images["status.webp"]; //横屏背景(左)
+				bgctx.drawImage(bg, 0, 0, BAR_WIDTH, GAMEVIEW_HEIGHT);
+				const bg2 = core.material.images.images["status.webp"]; //横屏背景(右)
+				bgctx.drawImage(
+					bg2,
+					BAR_WIDTH + GAMEVIEW_HEIGHT,
+					0,
+					BAR_WIDTH,
+					GAMEVIEW_HEIGHT
+				);
+				bgctx.globalAlpha = globalAlpha;
 
-        if (!floorId) return;
-        const info = core.plugin.getMapDrawInfo(floorId, Infinity, true);
-        core.setTextAlign("outerUI", "center");
+				bgctx.globalAlpha = 1;
+				core.setTextAlign("outerUI", "center");
+			}
+		}
+		// 更新属性
+		_update_props(updatedFloorTitle) {
+			if (!updatedFloorTitle && core.status.floorId) {
+				updatedFloorTitle = core.status.maps[core.status.floorId].title;
+			}
+			const statusList = [
+				"hp",
+				"atk",
+				"def",
+				"spell",
+				"mdef",
+				"matk",
+				"mhp",
+				"speed",
+				"money",
+				"exp"
+			]; //属性列表,图标在函数复写core.statusBar.icons中声明,数字为project\materials\icons.png中的图标序号(可使用便捷ps追加,第一个序号为0)
+			const drawStatusList = (baseX, baseY) => {
+				let curh = baseY;
+				core.setTextAlign("outerUI", "right");
+				statusList.forEach((item) => {
+					// 绘制图标
+					/*core.drawIcon(
+					  "outerUI",
+					  item,
+					  baseX - 95 * 3,
+					  curh - 18 * 3,
+					  22 * 3,
+					  22 * 3
+					);*/
+					core.strokeRect("outerUI", baseX - 96 * 3,
+						curh - 16 * 3, 100 * 3, 22 * 3, "#FFFFFF", 3)
+					core.setFont("outerUI", "bold 24px Verdana");
+					core.fillBoldText1(
+						"outerUI",
+						core.getStatusLabel(item),
+						baseX - 65 * 3,
+						curh - 4 * 3,
+						TEXT_COLOR,
+						"#000000",
+						3
+					);
+					core.setFont("outerUI", "bold 36px Verdana");
+					// 四舍五入
+					core.status.hero[item] = Math.round(core.status.hero[item]);
+					let text = core.getRealStatus(item);
+					// 大数据格式化
+					switch (item) {
+					case "mdef":
+						text += "%";
+						break;
+					case "matk":
+					case "mhp":
+						text = `${Math.floor(core.getRealStatus("spell")* core.getRealStatus(item)/100)}(${text}%)`;
+						break;
+					}
+					core.fillBoldText1(
+						"outerUI",
+						text,
+						baseX,
+						curh,
+						TEXT_COLOR,
+						"#000000",
+						3
+					)
 
-        core.plugin.drawSmallMap(uictx, info, floorId, x, y, 300, 300);
-      }
+					curh += 24 * 3;
+					if (curh > 130 * 3 && core.domStyle.isVertical) {
+						curh = 24 * 3;
+						baseX += 105 * 3;
+					}
+				});
+				core.setTextAlign("outerUI", "center");
+			};
+			if (core.domStyle.isVertical) {
+				core.clearMap("outerUI", 10 * 3, 0, 210 * 3, 120 * 3);
+				core.setFont("outerUI", "bold 42px Verdana");
+				if (updatedFloorTitle) {
+					core.fillBoldText1(
+						"outerUI",
+						updatedFloorTitle,
+						60 * 3,
+						22 * 3,
+						TEXT_COLOR,
+						"#000000",
+						3
+					);
+				}
+				drawStatusList(96 * 3, 46 * 3);
+				//core.drawImage("outerUI", "lane1.png", 0, 0)
+				//core.drawImage("outerUI", "cao.webp", 0, 0);
+			} else {
+				core.clearMap("outerUI", 10 * 3, 40 * 3, 105 * 3, 250 * 3);
+				core.setFont("outerUI", "bold 48px Verdana");
+				if (updatedFloorTitle) {
+					core.fillBoldText1(
+						"outerUI",
+						updatedFloorTitle,
+						62 * 3,
+						41 * 3,
+						TEXT_COLOR,
+						"#000000",
+						3
+					);
+				}
+				drawStatusList(110 * 3, 93 * 3);
+				//core.drawImage("outerUI", "lane1.png", 0, 30);
+				core.drawImage(
+					"outerUI",
+					"cao.webp",
+					0,
+					0,
+					400,
+					350,
+					0,
+					30,
+					360,
+					315
+				);
+			}
+		}
+		_update_items() {
+			//更新道具栏
+			const drawItemMx = (drawFn) => {
+				for (let i = 0; i < this.itemMx.length; i++) {
+					for (let j = 0; j < this.itemMx[i].length; j++) {
+						var item = this.itemMx[i][j];
+						drawFn(i, j, item);
+					}
+				}
+			};
+			const drawItem = (item, posx, posy) => {
+				const icon = core.material.icons.items[item],
+					image = core.material.images.items;
+				core.drawImage(
+					"outerUI",
+					image,
+					0,
+					32 * icon,
+					32,
+					32,
+					posx,
+					posy,
+					30 * 3,
+					30 * 3
+				);
+				const cnt = core.itemCount(item);
+				if (
+					(core.items.items[item].cls === "tools" && cnt > 1) ||
+					FORCE_COUNTABLE_ITEMS.includes(item)
+				) {
+					core.fillText(
+						"outerUI",
+						cnt,
+						posx + 25 * 3,
+						posy + 28 * 3,
+						"#FFFFFF",
+						"bold 36px Verdana"
+					);
+				}
+			};
+			if (core.domStyle.isVertical) {
+				core.clearMap(
+					"outerUI",
+					ITEM_BOX_LEFT_VERTICAL,
+					ITEM_BOX_TOP_VERTICAL,
+					185 * 3,
+					125 * 3
+				);
 
-      _update_equips() {
-        return;
-        core.setFont("outerUI", "bold 48px Verdana");
-        const drawEquip = (baseX, baseY, id, color, back) => {
-          if (!id)
-            core.fillText(
-              "outerUI",
-              back,
-              baseX + 20 * 3,
-              baseY + 22 * 3,
-              color
-            );
-          else {
-            var icon = core.material.icons.items[id];
-            core.drawImage(
-              "outerUI",
-              core.material.images.items,
-              0,
-              32 * icon,
-              32,
-              32,
-              baseX + 5 * 3,
-              baseY,
-              32 * 3,
-              32 * 3
-            );
-          }
-        };
-        if (core.domStyle.isVertical) {
-          core.clearMap(
-            "outerUI",
-            EQUIP_BLOCK_LEFT_VERTICAL,
-            EQUIP_BLOCK_TOP_VERTICAL,
-            90 * 3,
-            130 * 3
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT_VERTICAL,
-            EQUIP_BLOCK_TOP_VERTICAL,
-            core.getEquip(0),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT_VERTICAL + 45 * 3,
-            EQUIP_BLOCK_TOP_VERTICAL,
-            core.getEquip(1),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT_VERTICAL,
-            EQUIP_BLOCK_TOP_VERTICAL + 45 * 3,
-            core.getEquip(2),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT_VERTICAL + 45 * 3,
-            EQUIP_BLOCK_TOP_VERTICAL + 45 * 3,
-            core.getEquip(3),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT_VERTICAL,
-            EQUIP_BLOCK_TOP_VERTICAL + 90 * 3,
-            core.getEquip(4),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT_VERTICAL + 45 * 3,
-            EQUIP_BLOCK_TOP_VERTICAL + 90 * 3,
-            core.getEquip(5),
-            "#D1CEFF",
-            "无"
-          );
-        } else {
-          core.clearMap(
-            "outerUI",
-            EQUIP_BLOCK_LEFT,
-            EQUIP_BLOCK_TOP,
-            130 * 3,
-            95 * 3
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT,
-            EQUIP_BLOCK_TOP,
-            core.getEquip(0),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT + 42 * 3,
-            EQUIP_BLOCK_TOP,
-            core.getEquip(1),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT + 85 * 3,
-            EQUIP_BLOCK_TOP,
-            core.getEquip(2),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT,
-            EQUIP_BLOCK_TOP + 45 * 3,
-            core.getEquip(3),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT + 42 * 3,
-            EQUIP_BLOCK_TOP + 45 * 3,
-            core.getEquip(4),
-            "#D1CEFF",
-            "无"
-          );
-          drawEquip(
-            EQUIP_BLOCK_LEFT + 85 * 3,
-            EQUIP_BLOCK_TOP + 45 * 3,
-            core.getEquip(5),
-            "#D1CEFF",
-            "无"
-          );
-        }
-      }
-      _update_keys() {
-        const drawKeyList = (baseX, baseY) => {
-          const todraw = [],
-            keyList = ["yellowKey", "blueKey", "redKey", "greenKey"];
-          let total = 0;
-          keyList.forEach(function (key, i) {
-            todraw[i] = core.itemCount(key);
-            total += todraw[i];
-          });
+				drawItemMx((i, j, item) => {
+					if (core.hasItem(item)) {
+						const posx = ITEM_BOX_LEFT_VERTICAL + i * 30 * 3,
+							posy = ITEM_BOX_TOP_VERTICAL + j * 31 * 3;
+						drawItem(item, posx, posy);
+					}
+				});
+			} else {
+				core.clearMap(
+					"outerUI",
+					ITEM_BOX_LEFT,
+					ITEM_BOX_TOP,
+					125 * 3,
+					185 * 3
+				);
 
-          let dn = 3;
-          for (let i = 0; i <= dn; i++) {
-            let delta = i * 32 * 3;
+				drawItemMx((i, j, item) => {
+					if (core.hasItem(item)) {
+						const posx = ITEM_BOX_LEFT + j * 30 * 3,
+							posy = ITEM_BOX_TOP + i * 31 * 3;
+						drawItem(item, posx, posy);
+					}
+				});
+			}
+		}
 
-            if (core.domStyle.isVertical) {
-              this.drawKey(keyList[i], baseX, baseY + delta);
-            } else {
-              this.drawKey(keyList[i], baseX + delta, baseY);
-            }
+		_update_map(floorId = core.status.floorId) {
+			const x = core.domStyle.isVertical ?
+				MAP_BLOCK_LEFT_VERTICAL :
+				MAP_BLOCK_LEFT;
+			const y = core.domStyle.isVertical ?
+				MAP_BLOCK_TOP_VERTICAL :
+				MAP_BLOCK_TOP;
 
-            core.setFont("outerUI", "bold 48px Verdana");
-            core.setTextAlign("outerUI", "left");
-            if (core.domStyle.isVertical) {
-              core.fillText(
-                "outerUI",
-                todraw[i],
-                baseX + 20 * 3,
-                baseY + 14 * 3 + delta,
-                TEXT_COLOR
-              );
-            } else {
-              core.fillText(
-                "outerUI",
-                todraw[i],
-                baseX + delta,
-                baseY + 32 * 3,
-                TEXT_COLOR
-              );
-            }
-          }
-        };
-        if (core.domStyle.isVertical) {
-          core.clearMap(
-            "outerUI",
-            KEY_BLOCK_LEFT_VERTICAL,
-            KEY_BLOCK_TOP_VERTICAL,
-            45 * 3,
-            130 * 3
-          );
-          drawKeyList(
-            KEY_BLOCK_LEFT_VERTICAL + 3 * 3,
-            KEY_BLOCK_TOP_VERTICAL + 5 * 3
-          );
-        } else {
-          core.clearMap(
-            "outerUI",
-            KEY_BLOCK_LEFT,
-            KEY_BLOCK_TOP,
-            130 * 3,
-            45 * 3
-          );
-          drawKeyList(KEY_BLOCK_LEFT + 10 * 3, KEY_BLOCK_TOP);
-        }
-      }
-      drawKey(key, x, y) {
-        let sx = 0,
-          sy = 0;
+			if (!floorId) return;
+			const info = core.plugin.getMapDrawInfo(floorId, Infinity, true);
+			core.setTextAlign("outerUI", "center");
 
-        if (key == "yellowKey") sx += 13;
-        else if (key == "blueKey") sx += 26;
-        else if (key == "greenKey") sx += 39;
+			core.plugin.drawSmallMap(uictx, info, floorId, x, y, 300, 300);
+		}
 
-        core.drawImage(
-          "outerUI",
-          "maba.webp",
-          sx,
-          sy,
-          13,
-          26,
-          x,
-          y,
-          13 * 3,
-          26 * 3
-        );
-      }
-      _update_infoWindow() {
-        const itemId = this.selectedItem;
-        let text = "";
-        if (this.selectedItem) {
-          text = core.replaceText(core.material.items[itemId]?.text);
-          if (text[0] == "," || text[0] == ",") text = text.substring(1);
-        }
-        if (core.domStyle.isVertical) {
-          core.clearMap(
-            "outerUI",
-            INFO_BLOCK_LEFT_VERTICAL,
-            INFO_BLOCK_TOP_VERTICAL,
-            300 * 3,
-            120 * 3
-          );
+		_update_equips() {
+			return;
+			core.setFont("outerUI", "bold 48px Verdana");
+			const drawEquip = (baseX, baseY, id, color, back) => {
+				if (!id)
+					core.fillText(
+						"outerUI",
+						back,
+						baseX + 20 * 3,
+						baseY + 22 * 3,
+						color
+					);
+				else {
+					var icon = core.material.icons.items[id];
+					core.drawImage(
+						"outerUI",
+						core.material.images.items,
+						0,
+						32 * icon,
+						32,
+						32,
+						baseX + 5 * 3,
+						baseY,
+						32 * 3,
+						32 * 3
+					);
+				}
+			};
+			if (core.domStyle.isVertical) {
+				core.clearMap(
+					"outerUI",
+					EQUIP_BLOCK_LEFT_VERTICAL,
+					EQUIP_BLOCK_TOP_VERTICAL,
+					90 * 3,
+					130 * 3
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT_VERTICAL,
+					EQUIP_BLOCK_TOP_VERTICAL,
+					core.getEquip(0),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT_VERTICAL + 45 * 3,
+					EQUIP_BLOCK_TOP_VERTICAL,
+					core.getEquip(1),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT_VERTICAL,
+					EQUIP_BLOCK_TOP_VERTICAL + 45 * 3,
+					core.getEquip(2),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT_VERTICAL + 45 * 3,
+					EQUIP_BLOCK_TOP_VERTICAL + 45 * 3,
+					core.getEquip(3),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT_VERTICAL,
+					EQUIP_BLOCK_TOP_VERTICAL + 90 * 3,
+					core.getEquip(4),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT_VERTICAL + 45 * 3,
+					EQUIP_BLOCK_TOP_VERTICAL + 90 * 3,
+					core.getEquip(5),
+					"#D1CEFF",
+					"无"
+				);
+			} else {
+				core.clearMap(
+					"outerUI",
+					EQUIP_BLOCK_LEFT,
+					EQUIP_BLOCK_TOP,
+					130 * 3,
+					95 * 3
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT,
+					EQUIP_BLOCK_TOP,
+					core.getEquip(0),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT + 42 * 3,
+					EQUIP_BLOCK_TOP,
+					core.getEquip(1),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT + 85 * 3,
+					EQUIP_BLOCK_TOP,
+					core.getEquip(2),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT,
+					EQUIP_BLOCK_TOP + 45 * 3,
+					core.getEquip(3),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT + 42 * 3,
+					EQUIP_BLOCK_TOP + 45 * 3,
+					core.getEquip(4),
+					"#D1CEFF",
+					"无"
+				);
+				drawEquip(
+					EQUIP_BLOCK_LEFT + 85 * 3,
+					EQUIP_BLOCK_TOP + 45 * 3,
+					core.getEquip(5),
+					"#D1CEFF",
+					"无"
+				);
+			}
+		}
+		_update_keys() {
+			const drawKeyList = (baseX, baseY) => {
+				const todraw = [],
+					keyList = ["yellowKey", "blueKey", "redKey", "greenKey"];
+				let total = 0;
+				keyList.forEach(function (key, i) {
+					todraw[i] = core.itemCount(key);
+					total += todraw[i];
+				});
 
-          if (this.selectedItem) {
-            const icon = core.material.icons.items[itemId];
-            core.setTextAlign("outerUI", "left");
-            core.fillText(
-              "outerUI",
-              core.material.items[itemId].name,
-              INFO_BLOCK_LEFT_VERTICAL + 50 * 3,
-              INFO_BLOCK_TOP_VERTICAL + 27 * 3,
-              "#D1CEFF"
-            );
-            core.drawImage(
-              "outerUI",
-              core.material.images.items,
-              0,
-              32 * icon,
-              32,
-              32,
-              INFO_BLOCK_LEFT_VERTICAL + 10 * 3,
-              INFO_BLOCK_TOP_VERTICAL + 8 * 3,
-              32 * 3,
-              32 * 3
-            );
-            core.ui.drawTextContent("outerUI", text, {
-              left: INFO_BLOCK_LEFT_VERTICAL + 10 * 3,
-              top: INFO_BLOCK_TOP_VERTICAL + 40 * 3,
-              maxWidth: 275 * 3,
-              color: "#D1CEFF",
-              fontSize: 36,
-            });
-          }
-        } else {
-          core.clearMap(
-            "outerUI",
-            INFO_BLOCK_LEFT,
-            INFO_BLOCK_TOP,
-            115 * 3,
-            230 * 3
-          );
+				let dn = 3;
+				for (let i = 0; i <= dn; i++) {
+					let delta = i * 32 * 3;
 
-          if (this.selectedItem) {
-            const icon = core.material.icons.items[itemId];
-            core.setTextAlign("outerUI", "center");
-            core.fillText(
-              "outerUI",
-              core.material.items[itemId].name,
-              INFO_BLOCK_LEFT + 60 * 3,
-              INFO_BLOCK_TOP + 25 * 3,
-              "#D1CEFF"
-            );
-            core.drawImage(
-              "outerUI",
-              core.material.images.items,
-              0,
-              32 * icon,
-              32,
-              32,
-              INFO_BLOCK_LEFT + 45 * 3,
-              INFO_BLOCK_TOP + 30 * 3,
-              32 * 3,
-              32 * 3
-            );
-            core.ui.drawTextContent("outerUI", text, {
-              left: INFO_BLOCK_LEFT + 10 * 3,
-              top: INFO_BLOCK_TOP + 60 * 3,
-              maxWidth: 105 * 3,
-              color: "#D1CEFF",
-              fontSize: 36,
-            });
-          }
-        }
-      }
-      showItemInfo(itemId) {
-        //展示道具说明
-        this.selectedItem = itemId;
-        this._update_infoWindow();
-      }
-      clearItemInfo() {
-        //清除道具说明
-        this.selectedItem = null;
-        this._update_infoWindow();
-      }
-      _update_toolBox() {
-        const tools = core.isReplaying()
-          ? [
-              [core.status.replay.pausing ? "play" : "pause", "stop", "rewind"],
-              ["speedDown", "speedUp", "save"],
-            ]
-          : [
-              ["keyboard", "shop", "pack", "T332"],
-              ["settings", "save", "load", "T331"],
-            ];
-        if (core.domStyle.isVertical) {
-          core.clearMap(
-            "outerUI",
-            TOOL_BOX_LEFT_VERTICAL,
-            TOOL_BOX_TOP_VERTICAL,
-            115,
-            130
-          );
+					if (core.domStyle.isVertical) {
+						this.drawKey(keyList[i], baseX, baseY + delta);
+					} else {
+						this.drawKey(keyList[i], baseX + delta, baseY);
+					}
 
-          for (let i = 0; i < tools.length; i++) {
-            for (let j = 0; j < tools[i].length; j++) {
-              core.drawIcon(
-                "outerUI",
-                tools[i][j],
-                TOOL_BOX_LEFT_VERTICAL + i * 31 * 3,
-                TOOL_BOX_TOP_VERTICAL + j * 31 * 3,
-                30 * 3,
-                30 * 3
-              );
-            }
-          }
-        } else {
-          core.clearMap(
-            "outerUI",
-            TOOL_BOX_LEFT,
-            TOOL_BOX_TOP,
-            130 * 3,
-            80 * 3
-          );
+					core.setFont("outerUI", "bold 48px Verdana");
+					core.setTextAlign("outerUI", "left");
+					if (core.domStyle.isVertical) {
+						core.fillText(
+							"outerUI",
+							todraw[i],
+							baseX + 20 * 3,
+							baseY + 14 * 3 + delta,
+							TEXT_COLOR
+						);
+					} else {
+						core.fillText(
+							"outerUI",
+							todraw[i],
+							baseX + delta,
+							baseY + 32 * 3,
+							TEXT_COLOR
+						);
+					}
+				}
+			};
+			if (core.domStyle.isVertical) {
+				core.clearMap(
+					"outerUI",
+					KEY_BLOCK_LEFT_VERTICAL,
+					KEY_BLOCK_TOP_VERTICAL,
+					45 * 3,
+					130 * 3
+				);
+				drawKeyList(
+					KEY_BLOCK_LEFT_VERTICAL + 3 * 3,
+					KEY_BLOCK_TOP_VERTICAL + 5 * 3
+				);
+			} else {
+				core.clearMap(
+					"outerUI",
+					KEY_BLOCK_LEFT,
+					KEY_BLOCK_TOP,
+					130 * 3,
+					45 * 3
+				);
+				drawKeyList(KEY_BLOCK_LEFT + 10 * 3, KEY_BLOCK_TOP);
+			}
+		}
+		drawKey(key, x, y) {
+			let sx = 0,
+				sy = 0;
 
-          for (let i = 0; i < tools.length; i++) {
-            for (let j = 0; j < tools[i].length; j++) {
-              core.drawIcon(
-                "outerUI",
-                tools[i][j],
-                TOOL_BOX_LEFT + j * 31 * 3,
-                TOOL_BOX_TOP + i * 31 * 3,
-                30 * 3,
-                30 * 3
-              );
-            }
-          }
-        }
-      }
-      onclick(x, y) {
-        const makeBox = ([x, y], [w, h]) => {
-          return [
-            [x, y],
-            [x + w, y + h],
-          ];
-        };
-        const gridify = ([x, y], [gw, gh]) => {
-          return [Math.floor(x / gw), Math.floor(y / gh)];
-        };
-        const useItem = (itemId) => {
-          if (!core.hasItem(itemId)) return;
+			if (key == "yellowKey") sx += 13;
+			else if (key == "blueKey") sx += 26;
+			else if (key == "greenKey") sx += 39;
 
-          if (itemId != this.selectedItem) {
-            this.showItemInfo(itemId);
-          } else {
-            switch (itemId) {
-              case "centerFly":
-                core.ui._drawCenterFly();
-                break;
-              case "book":
-                core.openBook(true);
-                break;
-              case "wand":
-                core.insertAction({
-                  type: "useItem",
-                  id: itemId,
-                });
-                break;
-              case "fly":
-                core.useItem(itemId, true);
-                break;
-              default:
-                core.useItem(itemId);
-            }
-          }
-        };
-        const inRect = ([x, y], [[sx, sy], [dx, dy]]) => {
-          return sx <= x && x <= dx && sy <= y && y <= dy;
-        };
-        const relativeTo = ([x, y], [ax, ay]) => {
-          return [x - ax, y - ay];
-        };
-        const pos = [x, y];
-        if (core.domStyle.isVertical) {
-          const itemBox = makeBox(
-            [ITEM_BOX_LEFT_VERTICAL, ITEM_BOX_TOP_VERTICAL],
-            [30 * 6 * 3, 31 * 4 * 3]
-          );
-          if (inRect(pos, itemBox)) {
-            const [gx, gy] = gridify(relativeTo(pos, itemBox[0]), [
-              30 * 3,
-              31 * 3,
-            ]);
-            const itemId = this.itemMx[gx][gy];
-            if (
-              (core.status.event.id == "viewMaps" ||
-                core.status.event.id == "fly") &&
-              itemId === "book"
-            )
-              core.openBook(true);
-            if (
-              core.isReplaying() ||
-              core.status.lockControl ||
-              core.isMoving()
-            )
-              return;
-            useItem(itemId);
-            return;
-          }
-          const toolBox = makeBox(
-            [TOOL_BOX_LEFT_VERTICAL, TOOL_BOX_TOP_VERTICAL],
-            [31 * 2 * 3, 31 * 4 * 3]
-          );
-          if (inRect(pos, toolBox)) {
-            const [col, row] = gridify(relativeTo(pos, toolBox[0]), [
-              31 * 3,
-              31 * 3,
-            ]);
-            if (core.isReplaying()) {
-              this.replayAction[col][row].call(core);
-            } else if (core.isPlaying()) {
-              if (col === 0 && row === 3) {
-                core.doSL("autoSave", "load");
-              } else if (col === 1 && row === 3) {
-                core.doSL("autoSave", "reload");
-              } else {
-                this.toolbarAction[col][row].call(core, true);
-              }
-            }
-            return;
-          }
-          const mapBox = makeBox(
-            [MAP_BLOCK_LEFT_VERTICAL, MAP_BLOCK_TOP_VERTICAL],
-            [350, 350]
-          );
-          if (inRect(pos, mapBox)) {
-            if (
-              core.isReplaying() ||
-              core.status.lockControl ||
-              core.isMoving()
-            )
-              return;
-            core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
-            return;
-          }
-          /*const equipBox = makeBox([EQUIP_BLOCK_LEFT_VERTICAL, EQUIP_BLOCK_TOP_VERTICAL], [90 * 3, 130 * 3])
+			core.drawImage(
+				"outerUI",
+				"maba.webp",
+				sx,
+				sy,
+				13,
+				26,
+				x,
+				y,
+				13 * 3,
+				26 * 3
+			);
+		}
+		_update_infoWindow() {
+			const itemId = this.selectedItem;
+			let text = "";
+			if (this.selectedItem) {
+				text = core.replaceText(core.material.items[itemId]?.text);
+				if (text[0] == "," || text[0] == ",") text = text.substring(1);
+			}
+			if (core.domStyle.isVertical) {
+				core.clearMap(
+					"outerUI",
+					INFO_BLOCK_LEFT_VERTICAL,
+					INFO_BLOCK_TOP_VERTICAL,
+					300 * 3,
+					120 * 3
+				);
+
+				if (this.selectedItem) {
+					const icon = core.material.icons.items[itemId];
+					core.setTextAlign("outerUI", "left");
+					core.fillText(
+						"outerUI",
+						core.material.items[itemId].name,
+						INFO_BLOCK_LEFT_VERTICAL + 50 * 3,
+						INFO_BLOCK_TOP_VERTICAL + 27 * 3,
+						"#D1CEFF"
+					);
+					core.drawImage(
+						"outerUI",
+						core.material.images.items,
+						0,
+						32 * icon,
+						32,
+						32,
+						INFO_BLOCK_LEFT_VERTICAL + 10 * 3,
+						INFO_BLOCK_TOP_VERTICAL + 8 * 3,
+						32 * 3,
+						32 * 3
+					);
+					core.ui.drawTextContent("outerUI", text, {
+						left: INFO_BLOCK_LEFT_VERTICAL + 10 * 3,
+						top: INFO_BLOCK_TOP_VERTICAL + 40 * 3,
+						maxWidth: 275 * 3,
+						color: "#D1CEFF",
+						fontSize: 24,
+					});
+				}
+			} else {
+				core.clearMap(
+					"outerUI",
+					INFO_BLOCK_LEFT,
+					INFO_BLOCK_TOP,
+					115 * 3,
+					230 * 3
+				);
+				core.setFont("outerUI", "bold 36px Verdana")
+				if (this.selectedItem) {
+					const icon = core.material.icons.items[itemId];
+					core.setTextAlign("outerUI", "center");
+					core.fillText(
+						"outerUI",
+						core.material.items[itemId].name,
+						INFO_BLOCK_LEFT + 60 * 3,
+						INFO_BLOCK_TOP + 25 * 3,
+						"#D1CEFF"
+					);
+					core.drawImage(
+						"outerUI",
+						core.material.images.items,
+						0,
+						32 * icon,
+						32,
+						32,
+						INFO_BLOCK_LEFT + 45 * 3,
+						INFO_BLOCK_TOP + 30 * 3,
+						32 * 3,
+						32 * 3
+					);
+					core.ui.drawTextContent("outerUI", text, {
+						left: INFO_BLOCK_LEFT + 10 * 3,
+						top: INFO_BLOCK_TOP + 60 * 3,
+						maxWidth: 105 * 3,
+						color: "#D1CEFF",
+						fontSize: 24,
+					});
+				}
+			}
+		}
+		showItemInfo(itemId) {
+			//展示道具说明
+			this.selectedItem = itemId;
+			this._update_infoWindow();
+		}
+		clearItemInfo() {
+			//清除道具说明
+			this.selectedItem = null;
+			this._update_infoWindow();
+		}
+		_update_toolBox() {
+			const tools = core.isReplaying() ? [
+				[core.status.replay.pausing ? "play" : "pause", "stop", "rewind"],
+				["speedDown", "speedUp", "save"],
+			] : [
+				["keyboard", "shop", "pack", "T332"],
+				["save", "load", "settings", "T331"],
+			];
+			if (core.domStyle.isVertical) {
+				core.clearMap(
+					"outerUI",
+					TOOL_BOX_LEFT_VERTICAL,
+					TOOL_BOX_TOP_VERTICAL,
+					115,
+					130
+				);
+
+				for (let i = 0; i < tools.length; i++) {
+					for (let j = 0; j < tools[i].length; j++) {
+						core.drawIcon(
+							"outerUI",
+							tools[i][j],
+							TOOL_BOX_LEFT_VERTICAL + i * 31 * 3,
+							TOOL_BOX_TOP_VERTICAL + j * 31 * 3,
+							30 * 3,
+							30 * 3
+						);
+					}
+				}
+			} else {
+				core.clearMap(
+					"outerUI",
+					TOOL_BOX_LEFT,
+					TOOL_BOX_TOP,
+					130 * 3,
+					80 * 3
+				);
+
+				for (let i = 0; i < tools.length; i++) {
+					for (let j = 0; j < tools[i].length; j++) {
+						core.drawIcon(
+							"outerUI",
+							tools[i][j],
+							TOOL_BOX_LEFT + j * 31 * 3,
+							TOOL_BOX_TOP + i * 31 * 3,
+							30 * 3,
+							30 * 3
+						);
+					}
+				}
+			}
+		}
+		onclick(x, y) {
+			const makeBox = ([x, y], [w, h]) => {
+				return [
+					[x, y],
+					[x + w, y + h],
+				];
+			};
+			const gridify = ([x, y], [gw, gh]) => {
+				return [Math.floor(x / gw), Math.floor(y / gh)];
+			};
+			const useItem = (itemId) => {
+				if (!core.hasItem(itemId)) return;
+
+				if (itemId != this.selectedItem) {
+					this.showItemInfo(itemId);
+				} else {
+					switch (itemId) {
+					case "centerFly":
+						core.ui._drawCenterFly();
+						break;
+					case "book":
+						core.openBook(true);
+						break;
+					default:
+						core.useItem(itemId);
+					}
+				}
+			};
+			const inRect = ([x, y], [
+				[sx, sy],
+				[dx, dy]
+			]) => {
+				return sx <= x && x <= dx && sy <= y && y <= dy;
+			};
+			const relativeTo = ([x, y], [ax, ay]) => {
+				return [x - ax, y - ay];
+			};
+			const pos = [x, y];
+			if (core.domStyle.isVertical) {
+				const itemBox = makeBox(
+					[ITEM_BOX_LEFT_VERTICAL, ITEM_BOX_TOP_VERTICAL],
+					[30 * 6 * 3, 31 * 4 * 3]
+				);
+				if (inRect(pos, itemBox)) {
+					const [gx, gy] = gridify(relativeTo(pos, itemBox[0]), [
+						30 * 3,
+						31 * 3,
+					]);
+					const itemId = this.itemMx[gx][gy];
+					if (
+						(core.status.event.id == "viewMaps" ||
+							core.status.event.id == "fly") &&
+						itemId === "book"
+					)
+						core.openBook(true);
+					if (
+						core.isReplaying() ||
+						core.status.lockControl ||
+						core.isMoving()
+					)
+						return;
+					useItem(itemId);
+					return;
+				}
+				const toolBox = makeBox(
+					[TOOL_BOX_LEFT_VERTICAL, TOOL_BOX_TOP_VERTICAL],
+					[31 * 2 * 3, 31 * 4 * 3]
+				);
+				if (inRect(pos, toolBox)) {
+					const [col, row] = gridify(relativeTo(pos, toolBox[0]), [
+						31 * 3,
+						31 * 3,
+					]);
+					if (core.status.lockControl || core.isMoving()) return;
+					if (core.isReplaying()) {
+						this.replayAction[col][row].call(core);
+					} else if (core.isPlaying()) {
+						if (col === 0 && row === 3) {
+							core.doSL("autoSave", "load");
+						} else if (col === 1 && row === 3) {
+							core.doSL("autoSave", "reload");
+						} else {
+							this.toolbarAction[col][row].call(core, true);
+						}
+					}
+					return;
+				}
+				const mapBox = makeBox(
+					[MAP_BLOCK_LEFT_VERTICAL, MAP_BLOCK_TOP_VERTICAL],
+					[350, 350]
+				);
+				if (inRect(pos, mapBox)) {
+					if (
+						core.isReplaying() ||
+						core.status.lockControl ||
+						core.isMoving()
+					)
+						return;
+					core.useItem("fly");
+					return;
+				}
+				/*const equipBox = makeBox([EQUIP_BLOCK_LEFT_VERTICAL, EQUIP_BLOCK_TOP_VERTICAL], [90 * 3, 130 * 3])
 				if (inRect(pos, equipBox)) {
 					if (core.isReplaying() || core.status.lockControl || core.isMoving()) return;
 					core.openEquipbox(true)
 					return;
 				}*/
-        } else {
-          const mapBox = makeBox([MAP_BLOCK_LEFT, MAP_BLOCK_TOP], [350, 350]);
-          if (inRect(pos, mapBox)) {
-            if (
-              core.isReplaying() ||
-              core.status.lockControl ||
-              core.isMoving()
-            )
-              return;
-            core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
-            return;
-          }
-          /*
+			} else {
+				const mapBox = makeBox([MAP_BLOCK_LEFT, MAP_BLOCK_TOP], [350, 350]);
+				if (inRect(pos, mapBox)) {
+					if (
+						core.isReplaying() ||
+						core.status.lockControl ||
+						core.isMoving()
+					)
+						return;
+					core.useItem("fly");
+					return;
+				}
+				/*
 								const equipBox = makeBox([EQUIP_BLOCK_LEFT, EQUIP_BLOCK_TOP], [130, 95])
 								if (inRect(pos, equipBox)) {
 									if (core.isReplaying() || core.status.lockControl || core.isMoving()) return;
 									core.openEquipbox(true)
 									return;
 								}*/
-          const itemBox = makeBox(
-            [ITEM_BOX_LEFT, ITEM_BOX_TOP],
-            [31 * 4 * 3, 30 * 6 * 3]
-          );
-          if (inRect(pos, itemBox)) {
-            const [gx, gy] = gridify(relativeTo(pos, itemBox[0]), [
-              31 * 3,
-              30 * 3,
-            ]);
-            const itemId = this.itemMx[gy][gx];
-            if (
-              (core.status.event.id == "viewMaps" ||
-                core.status.event.id == "fly") &&
-              itemId === "book"
-            )
-              core.openBook(true);
-            if (
-              core.isReplaying() ||
-              core.status.lockControl ||
-              core.isMoving()
-            )
-              return;
-            useItem(itemId);
-            return;
-          }
-          const toolBox = makeBox(
-            [TOOL_BOX_LEFT, TOOL_BOX_TOP],
-            [31 * 4 * 3, 31 * 2 * 3]
-          );
-          if (inRect(pos, toolBox)) {
-            const [row, col] = gridify(relativeTo(pos, toolBox[0]), [
-              31 * 3,
-              31 * 3,
-            ]);
-            if (core.isReplaying()) {
-              this.replayAction[col][row].call(core);
-            } else if (core.isPlaying()) {
-              if (col === 0 && row === 3) {
-                core.doSL("autoSave", "load");
-              } else if (col === 1 && row === 3) {
-                core.doSL("autoSave", "reload");
-              } else {
-                this.toolbarAction[col][row].call(core, true);
-              }
-            }
-            return;
-          }
-        }
-      }
-    }
+				const itemBox = makeBox(
+					[ITEM_BOX_LEFT, ITEM_BOX_TOP],
+					[31 * 4 * 3, 30 * 6 * 3]
+				);
+				if (inRect(pos, itemBox)) {
+					const [gx, gy] = gridify(relativeTo(pos, itemBox[0]), [
+						31 * 3,
+						30 * 3,
+					]);
+					const itemId = this.itemMx[gy][gx];
+					if (
+						(core.status.event.id == "viewMaps" ||
+							core.status.event.id == "fly") &&
+						itemId === "book"
+					)
+						core.openBook(true);
+					if (
+						core.isReplaying() ||
+						core.status.lockControl ||
+						core.isMoving()
+					)
+						return;
+					useItem(itemId);
+					return;
+				}
+				const toolBox = makeBox(
+					[TOOL_BOX_LEFT, TOOL_BOX_TOP],
+					[31 * 4 * 3, 31 * 2 * 3]
+				);
+				if (inRect(pos, toolBox)) {
+					const [row, col] = gridify(relativeTo(pos, toolBox[0]), [
+						31 * 3,
+						31 * 3,
+					]);
+					if (core.status.lockControl || core.isMoving()) return;
+					if (core.isReplaying()) {
+						this.replayAction[col][row].call(core);
+					} else if (core.isPlaying()) {
+						if (col === 0 && row === 3) {
+							core.doSL("autoSave", "load");
+						} else if (col === 1 && row === 3) {
+							core.doSL("autoSave", "reload");
+						} else {
+							this.toolbarAction[col][row].call(core, true);
+						}
+					}
+					return;
+				}
+			}
+		}
+	}
 
-    core.ui.statusBar = new StatusBar();
+	core.ui.statusBar = new StatusBar();
 
-    core.control.clearStatusBar = function () {
-      core.clearMap("outerUI");
-    };
-    // init() called in `afterLoadResources`.
-  },
+	core.control.clearStatusBar = function () {
+		core.clearMap("outerUI");
+	};
+	// init() called in `afterLoadResources`.
+},
     "override": function () {
-    core.statusBar.icons = {
-      floor: 0,
-      name: null,
-      lv: 1,
-      hpmax: 2,
-      hp: 3,
-      atk: 4,
-      def: 5,
-      mdef: 6,
-      money: 7,
-      exp: 8,
-      up: 9,
-      book: 10,
-      fly: 11,
-      toolbox: 12,
-      keyboard: 13,
-      shop: 14,
-      save: 15,
-      load: 16,
-      settings: 17,
-      play: 18,
-      pause: 19,
-      stop: 20,
-      speedDown: 21,
-      speedUp: 22,
-      rewind: 23,
-      equipbox: 24,
-      mana: 25,
-      skill: 26,
-      exit: 27,
-      btn1: 28,
-      btn2: 29,
-      btn3: 30,
-      btn4: 31,
-      btn5: 32,
-      btn6: 33,
-      btn7: 34,
-      alt: 35,
-      keys: 36,
-      help: 37,
-      battle: 38,
-    };
-    core.actions._getClickLoc = function (x, y) {
-      var size = 32 * core.domStyle.scale;
-      var left = main.dom.gameDraw.offsetLeft + main.dom.gameGroup.offsetLeft;
-      var top = main.dom.gameDraw.offsetTop + main.dom.gameGroup.offsetTop;
-      var loc = {
-        x: Math.max(x - left, 0),
-        y: Math.max(y - top, 0),
-        size: size,
-      };
-      return loc;
-    };
-    core.ui._drawWindowSelector = function (background, x, y, w, h) {
-      w = Math.round(w) + 48;
-      h = Math.round(h);
-      var ctx = core.ui.createCanvas("_selector", x - 24, y, w, h, 165);
-      ctx.canvas.id = "";
-      this._drawSelector(ctx, background, w, h);
-    };
+	core.statusBar.icons = {
+		floor: 0,
+		name: null,
+		lv: 1,
+		hpmax: 2,
+		hp: 3,
+		atk: 4,
+		def: 5,
+		mdef: 6,
+		money: 7,
+		exp: 8,
+		up: 9,
+		book: 10,
+		fly: 11,
+		toolbox: 12,
+		keyboard: 13,
+		shop: 14,
+		save: 15,
+		load: 16,
+		settings: 17,
+		play: 18,
+		pause: 19,
+		stop: 20,
+		speedDown: 21,
+		speedUp: 22,
+		rewind: 23,
+		equipbox: 24,
+		mana: 25,
+		skill: 26,
+		exit: 27,
+		btn1: 28,
+		btn2: 29,
+		btn3: 30,
+		btn4: 31,
+		btn5: 32,
+		btn6: 33,
+		btn7: 34,
+		alt: 35,
+		keys: 36,
+		help: 37,
+		battle: 38,
+	};
+	core.actions._getClickLoc = function (x, y) {
+		var size = 32 * core.domStyle.scale;
+		var left = main.dom.gameDraw.offsetLeft + main.dom.gameGroup.offsetLeft;
+		var top = main.dom.gameDraw.offsetTop + main.dom.gameGroup.offsetTop;
+		var loc = {
+			x: Math.max(x - left, 0),
+			y: Math.max(y - top, 0),
+			size: size,
+		};
+		return loc;
+	};
+	/* core.ui._drawWindowSelector = function (background, x, y, w, h) {
+	   w = Math.round(w) + 48;
+	   h = Math.round(h);
+	   var ctx = core.ui.createCanvas("_selector", x - 24, y, w, h, 165);
+	   ctx.canvas.id = "";
+	   this._drawSelector(ctx, background, w, h);
+	 };
 
-    core.ui._drawSelector = function (ctx, background, w, h, left, top) {
-      left = left || 0;
-      top = top || 0;
-      ctx = this.getContextByName(ctx);
-      if (!ctx) return;
-      if (typeof background == "string")
-        background = core.material.images.images[background];
-      if (!(background instanceof Image)) return;
-      // badge
-      ctx.drawImage(background, 132, 68, 24, 24, left + 4, top + 4, 24, 24);
-      ctx.drawImage(
-        background,
-        132,
-        68,
-        24,
-        24,
-        w - left - 28,
-        top + 4,
-        24,
-        24
-      );
-    };
+	 core.ui._drawSelector = function (ctx, background, w, h, left, top) {
+	   left = left || 0;
+	   top = top || 0;
+	   ctx = this.getContextByName(ctx);
+	   if (!ctx) return;
+	   if (typeof background == "string")
+	     background = core.material.images.images[background];
+	   if (!(background instanceof Image)) return;
+	   // badge
+	   ctx.drawImage(background, 132, 68, 24, 24, left + 4, top + 4, 24, 24);
+	   ctx.drawImage(
+	     background,
+	     132,
+	     68,
+	     24,
+	     24,
+	     w - left - 28,
+	     top + 4,
+	     24,
+	     24
+	   );
+	 };*/
 
-    enemys.prototype._nextCriticals_useBinarySearch = function (
-      enemy,
-      info,
-      number,
-      x,
-      y,
-      floorId
-    ) {
-      var mon_hp = info.mon_hp,
-        hero_atk = core.status.hero.atk,
-        mon_def = info.mon_def,
-        pre = info.damage;
-      var list = [];
-      var start_atk = hero_atk;
-      if (info.__over__) {
-        start_atk += info.__overAtk__;
-        list.push([info.__overAtk__, -info.damage]);
-      }
-      var calNext = function (currAtk, maxAtk) {
-        var start = Math.floor(currAtk),
-          end = Math.floor(maxAtk);
-        if (start > end) return null;
+	enemys.prototype._nextCriticals_useBinarySearch = function (
+		enemy,
+		info,
+		number,
+		x,
+		y,
+		floorId
+	) {
+		var mon_hp = info.mon_hp,
+			hero_atk = core.status.hero.atk,
+			mon_def = info.mon_def,
+			pre = info.damage;
+		var list = [];
+		var start_atk = hero_atk;
+		if (info.__over__) {
+			start_atk += info.__overAtk__;
+			list.push([info.__overAtk__, -info.damage]);
+		}
+		var calNext = function (currAtk, maxAtk) {
+			var start = Math.floor(currAtk),
+				end = Math.floor(maxAtk);
+			if (start > end) return null;
 
-        while (start < end) {
-          var mid = Math.floor((start + end) / 2);
-          if (mid - start > end - mid) mid--;
-          var nextInfo = core.enemys.getDamageInfo(
-            enemy,
-            { atk: mid },
-            x,
-            y,
-            floorId
-          );
-          if (nextInfo == null || typeof nextInfo == "number") return null;
-          if (pre > nextInfo.damage) end = mid;
-          else start = mid + 1;
-        }
-        var nextInfo = core.enemys.getDamageInfo(
-          enemy,
-          { atk: start },
-          x,
-          y,
-          floorId
-        );
-        return nextInfo == null ||
-          typeof nextInfo == "number" ||
-          nextInfo.damage >= pre
-          ? null
-          : [start, nextInfo.damage];
-      };
-      var currAtk = start_atk;
-      while (true) {
-        var next = calNext(currAtk + 1, Number.MAX_SAFE_INTEGER, pre);
-        if (next == null) break;
-        currAtk = next[0];
-        pre = next[1];
-        list.push([currAtk - hero_atk, info.damage - pre]);
-        if (pre <= 0 && !core.flags.enableNegativeDamage) break;
-        if (list.length >= number) break;
-      }
-      if (list.length == 0) list.push([0, 0]);
-      return list;
-    };
-    core.ui.clearMap = function (name, x, y, width, height) {
-      if (name == "all") {
-        for (var m in core.canvas) {
-          core.canvas[m].clearRect(
-            -32,
-            -32,
-            core.canvas[m].canvas.width + 32,
-            core.canvas[m].canvas.height + 32
-          );
-        }
-        core.clearMap("outerUI");
-        core.dom.gif.innerHTML = "";
-        core.removeGlobalAnimate();
-        core.deleteCanvas(function (one) {
-          return one.startsWith("_bigImage_");
-        });
-        core.setWeather(null);
-      } else {
-        var ctx = this.getContextByName(name);
-        if (ctx)
-          ctx.clearRect(
-            x || 0,
-            y || 0,
-            width || ctx.canvas.width,
-            height || ctx.canvas.height
-          );
-      }
-    };
-    events.prototype.openBook = function (fromUserAction) {
-      if (core.isReplaying()) return;
-      // 如果能恢复事件(从callBook事件触发)
-      if (
-        core.status.event.id == "book" &&
-        core.events.recoverEvents(core.status.event.interval)
-      )
-        return;
-      // 当前是book,且从“浏览地图”打开
-      if (core.status.event.id == "book" && core.status.event.ui) {
-        core.status.boxAnimateObjs = [];
-        core.ui._drawViewMaps(core.status.event.ui);
-        return;
-      }
-      // 从“浏览地图”页面打开
-      if (core.status.event.id == "viewMaps" || core.status.event.id == "fly") {
-        fromUserAction = false;
-        core.status.event.ui = core.status.event.data;
-      }
-      if (!this._checkStatus("book", fromUserAction, true)) return;
-      core.playSound("打开界面");
-      core.useItem("book", true);
-    };
-    ////// 怪物手册界面时,放开某个键的操作 //////
-    core.actions._keyUpBook = function (keycode) {
-      if (keycode == 27 || keycode == 88) {
-        core.playSound("取消");
-        if (core.events.recoverEvents(core.status.event.interval)) {
-          return;
-        } else if (core.status.event.ui != null) {
-          core.status.boxAnimateObjs = [];
-          if (typeof core.status.event.ui === "number") {
-            core.status.event.id = "fly";
-            core.ui.drawFly(core.status.event.ui);
-          } else {
-            core.ui._drawViewMaps(core.status.event.ui);
-          }
-        } else core.ui.closePanel();
-        return;
-      }
-      if (keycode == 13 || keycode == 32 || keycode == 67) {
-        var data = core.status.event.data;
-        if (data != null) {
-          core.ui._drawBookDetail(data);
-        }
-        return;
-      }
-    };
-    ////// 怪物手册界面的点击操作 //////
-    actions.prototype._clickBook = function (x, y) {
-      var pageinfo = core.ui._drawBook_pageinfo();
-      // 上一页
-      if (
-        (x == this._HX_ - 2 || x == this._HX_ - 3) &&
-        y === core._HEIGHT_ - 1
-      ) {
-        core.playSound("光标移动");
-        core.ui.drawBook(core.status.event.data - pageinfo.per_page);
-        return;
-      }
-      // 下一页
-      if (
-        (x == this._HX_ + 2 || x == this._HX_ + 3) &&
-        y === core._HEIGHT_ - 1
-      ) {
-        core.playSound("光标移动");
-        core.ui.drawBook(core.status.event.data + pageinfo.per_page);
-        return;
-      }
-      // 返回
-      if (x >= this.LAST - 2 && y === core._HEIGHT_ - 1) {
-        core.playSound("取消");
-        if (core.events.recoverEvents(core.status.event.interval)) {
-          return;
-        } else if (core.status.event.ui != null) {
-          core.status.boxAnimateObjs = [];
-          if (typeof core.status.event.ui === "number") {
-            core.status.event.id = "fly";
-            core.ui.drawFly(core.status.event.ui);
-          } else {
-            core.ui._drawViewMaps(core.status.event.ui);
-          }
-        } else core.ui.closePanel();
-        return;
-      }
-      // 怪物信息
-      var data = core.status.event.data;
-      if (data != null && y < core._HEIGHT_ - 1) {
-        var per_page = pageinfo.per_page,
-          page = parseInt(data / per_page);
-        var u = (core._HEIGHT_ - 1) / per_page;
-        for (var i = 0; i < per_page; ++i) {
-          if (y >= u * i && y < u * (i + 1)) {
-            var index = per_page * page + i;
-            core.ui.drawBook(index);
-            core.ui._drawBookDetail(index);
-            break;
-          }
-        }
-        return;
-      }
-      return;
-    };
+			while (start < end) {
+				var mid = Math.floor((start + end) / 2);
+				if (mid - start > end - mid) mid--;
+				var nextInfo = core.enemys.getDamageInfo(
+					enemy, { atk: mid },
+					x,
+					y,
+					floorId
+				);
+				if (nextInfo == null || typeof nextInfo == "number") return null;
+				if (pre > nextInfo.damage) end = mid;
+				else start = mid + 1;
+			}
+			var nextInfo = core.enemys.getDamageInfo(
+				enemy, { atk: start },
+				x,
+				y,
+				floorId
+			);
+			return nextInfo == null ||
+				typeof nextInfo == "number" ||
+				nextInfo.damage >= pre ?
+				null : [start, nextInfo.damage];
+		};
+		var currAtk = start_atk;
+		while (true) {
+			var next = calNext(currAtk + 1, Number.MAX_SAFE_INTEGER, pre);
+			if (next == null) break;
+			currAtk = next[0];
+			pre = next[1];
+			list.push([currAtk - hero_atk, info.damage - pre]);
+			if (pre <= 0 && !core.flags.enableNegativeDamage) break;
+			if (list.length >= number) break;
+		}
+		if (list.length == 0) list.push([0, 0]);
+		return list;
+	};
+	core.ui.clearMap = function (name, x, y, width, height) {
+		if (name == "all") {
+			for (var m in core.canvas) {
+				core.canvas[m].clearRect(
+					-32,
+					-32,
+					core.canvas[m].canvas.width + 32,
+					core.canvas[m].canvas.height + 32
+				);
+			}
+			core.clearMap("outerUI");
+			core.dom.gif.innerHTML = "";
+			core.removeGlobalAnimate();
+			core.deleteCanvas(function (one) {
+				return one.startsWith("_bigImage_");
+			});
+			core.setWeather(null);
+		} else {
+			var ctx = this.getContextByName(name);
+			if (ctx)
+				ctx.clearRect(
+					x || 0,
+					y || 0,
+					width || ctx.canvas.width,
+					height || ctx.canvas.height
+				);
+		}
+	};
+	events.prototype.openBook = function (fromUserAction) {
+		if (core.isReplaying()) return;
+		// 如果能恢复事件(从callBook事件触发)
+		if (
+			core.status.event.id == "book" &&
+			core.events.recoverEvents(core.status.event.interval)
+		)
+			return;
+		// 当前是book,且从“浏览地图”打开
+		if (core.status.event.id == "book" && core.status.event.ui) {
+			core.status.boxAnimateObjs = [];
+			core.ui._drawViewMaps(core.status.event.ui);
+			return;
+		}
+		// 从“浏览地图”页面打开
+		if (core.status.event.id == "viewMaps" || core.status.event.id == "fly") {
+			fromUserAction = false;
+			core.status.event.ui = core.status.event.data;
+		}
+		if (!this._checkStatus("book", fromUserAction, true)) return;
+		core.playSound("打开界面");
+		core.useItem("book", true);
+	};
+	////// 怪物手册界面时,放开某个键的操作 //////
+	core.actions._keyUpBook = function (keycode) {
+		if (keycode == 27 || keycode == 88) {
+			core.playSound("取消");
+			if (core.events.recoverEvents(core.status.event.interval)) {
+				return;
+			} else if (core.status.event.ui != null) {
+				core.status.boxAnimateObjs = [];
+				if (typeof core.status.event.ui === "number") {
+					core.status.event.id = "fly";
+					core.ui.drawFly(core.status.event.ui);
+				} else {
+					core.ui._drawViewMaps(core.status.event.ui);
+				}
+			} else core.ui.closePanel();
+			return;
+		}
+		if (keycode == 13 || keycode == 32 || keycode == 67) {
+			var data = core.status.event.data;
+			if (data != null) {
+				core.ui._drawBookDetail(data);
+			}
+			return;
+		}
+	};
+	////// 怪物手册界面的点击操作 //////
+	actions.prototype._clickBook = function (x, y) {
+		var pageinfo = core.ui._drawBook_pageinfo();
+		// 上一页
+		if (
+			(x == this._HX_ - 2 || x == this._HX_ - 3) &&
+			y === core._HEIGHT_ - 1
+		) {
+			core.playSound("光标移动");
+			core.ui.drawBook(core.status.event.data - pageinfo.per_page);
+			return;
+		}
+		// 下一页
+		if (
+			(x == this._HX_ + 2 || x == this._HX_ + 3) &&
+			y === core._HEIGHT_ - 1
+		) {
+			core.playSound("光标移动");
+			core.ui.drawBook(core.status.event.data + pageinfo.per_page);
+			return;
+		}
+		// 返回
+		if (x >= this.LAST - 2 && y === core._HEIGHT_ - 1) {
+			core.playSound("取消");
+			if (core.events.recoverEvents(core.status.event.interval)) {
+				return;
+			} else if (core.status.event.ui != null) {
+				core.status.boxAnimateObjs = [];
+				if (typeof core.status.event.ui === "number") {
+					core.status.event.id = "fly";
+					core.ui.drawFly(core.status.event.ui);
+				} else {
+					core.ui._drawViewMaps(core.status.event.ui);
+				}
+			} else core.ui.closePanel();
+			return;
+		}
+		// 怪物信息
+		var data = core.status.event.data;
+		if (data != null && y < core._HEIGHT_ - 1) {
+			var per_page = pageinfo.per_page,
+				page = parseInt(data / per_page);
+			var u = (core._HEIGHT_ - 1) / per_page;
+			for (var i = 0; i < per_page; ++i) {
+				if (y >= u * i && y < u * (i + 1)) {
+					var index = per_page * page + i;
+					core.ui.drawBook(index);
+					core.ui._drawBookDetail(index);
+					break;
+				}
+			}
+			return;
+		}
+		return;
+	};
 
-    ////// 执行当前自定义事件列表中的下一个事件 //////
-    events.prototype.doAction = function () {
-      // 清空boxAnimate和UI层
-      clearInterval(core.status.event.interval);
-      clearTimeout(core.status.event.interval);
-      clearInterval(core.status.event.animateUI);
-      core.status.event.interval = null;
-      delete core.status.event.aniamteUI;
-      if (core.status.gameOver || core.status.replay.failed) return;
-      // 判定是否执行完毕
-      if (this._doAction_finishEvents()) return;
-      core.clearUI();
-      var floorId = core.status.event.data.floorId || core.status.floorId;
-      // 当前点坐标和前缀
-      var x = core.status.event.data.x,
-        y = core.status.event.data.y;
-      var prefix = [
-        floorId || ":f",
-        x != null ? x : "x",
-        y != null ? y : "y",
-      ].join("@");
-      var current = core.status.event.data.list[0];
-      if (this._popEvents(current, prefix)) return;
-      // 当前要执行的事件
-      var data = current.todo.shift();
-      core.status.event.data.current = data;
-      if (typeof data == "string") data = { type: "text", text: data };
-      // 该事件块已经被禁用
-      if (data._disabled) return core.doAction();
-      if (data.type !== "cgtext") {
-        core.unregisterAnimationFrame("skip");
-        core.setFlag("skip", false);
-      }
-      data.floorId = data.floorId || floorId;
-      core.status.event.data.type = data.type;
-      this.doEvent(data, x, y, prefix);
-      return;
-    };
+	////// 执行当前自定义事件列表中的下一个事件 //////
+	events.prototype.doAction = function () {
+		// 清空boxAnimate和UI层
+		clearInterval(core.status.event.interval);
+		clearTimeout(core.status.event.interval);
+		clearInterval(core.status.event.animateUI);
+		core.status.event.interval = null;
+		delete core.status.event.aniamteUI;
+		if (core.status.gameOver || core.status.replay.failed) return;
+		// 判定是否执行完毕
+		if (this._doAction_finishEvents()) return;
+		core.clearUI();
+		var floorId = core.status.event.data.floorId || core.status.floorId;
+		// 当前点坐标和前缀
+		var x = core.status.event.data.x,
+			y = core.status.event.data.y;
+		var prefix = [
+			floorId || ":f",
+			x != null ? x : "x",
+			y != null ? y : "y",
+		].join("@");
+		var current = core.status.event.data.list[0];
+		if (this._popEvents(current, prefix)) return;
+		// 当前要执行的事件
+		var data = current.todo.shift();
+		core.status.event.data.current = data;
+		if (typeof data == "string") data = { type: "text", text: data };
+		// 该事件块已经被禁用
+		if (data._disabled) return core.doAction();
+		if (data.type !== "cgtext") {
+			core.unregisterAnimationFrame("skip");
+			core.setFlag("skip", false);
+		}
+		data.floorId = data.floorId || floorId;
+		core.status.event.data.type = data.type;
+		this.doEvent(data, x, y, prefix);
+		return;
+	};
 
-    ////// 在某个canvas上绘制粗体 //////
-    core.fillBoldText1 = function (
-      name,
-      text,
-      x,
-      y,
-      style,
-      strokeStyle,
-      lineWidth,
-      font,
-      maxWidth
-    ) {
-      var ctx = this.getContextByName(name);
-      if (!ctx) return;
-      if (font) ctx.font = font;
-      if (!style) style = ctx.fillStyle;
-      style = core.arrayToRGBA(style);
-      if (!strokeStyle) strokeStyle = "#000000";
-      strokeStyle = core.arrayToRGBA(strokeStyle);
-      if (maxWidth != null) {
-        this.setFontForMaxWidth(ctx, text, maxWidth);
-      }
-      ctx.strokeStyle = strokeStyle;
+	////// 在某个canvas上绘制粗体 //////
+	core.fillBoldText1 = function (
+		name,
+		text,
+		x,
+		y,
+		style,
+		strokeStyle,
+		lineWidth,
+		font,
+		maxWidth
+	) {
+		var ctx = this.getContextByName(name);
+		if (!ctx) return;
+		if (font) ctx.font = font;
+		if (!style) style = ctx.fillStyle;
+		style = core.arrayToRGBA(style);
+		if (!strokeStyle) strokeStyle = "#000000";
+		strokeStyle = core.arrayToRGBA(strokeStyle);
+		if (maxWidth != null) {
+			this.setFontForMaxWidth(ctx, text, maxWidth);
+		}
+		ctx.strokeStyle = strokeStyle;
 
-      if (!lineWidth) lineWidth = 2;
-      ctx.lineWidth = lineWidth;
-      ctx.strokeText(text, x, y);
-      ctx.fillStyle = style;
-      ctx.fillText(text, x, y);
-    };
-  },
+		if (!lineWidth) lineWidth = 2;
+		ctx.lineWidth = lineWidth;
+		ctx.strokeText(text, x, y);
+		ctx.fillStyle = style;
+		ctx.fillText(text, x, y);
+	};
+	////// 绘制 WindowSkin
+	ui.prototype.drawWindowSkin = function (
+		background,
+		ctx,
+		x,
+		y,
+		w,
+		h,
+		direction,
+		px,
+		py,
+		size = 1
+	) {
+		background = background || core.status.textAttribute.background;
+
+		// 仿RM窗口皮肤 ↓
+		// 绘制背景
+		core.drawImage(
+			ctx,
+			background,
+			0,
+			0,
+			128,
+			128,
+			x + 2 * size,
+			y + 2 * size,
+			w - 4 * size,
+			h - 4 * size
+		);
+		// 绘制边框
+		// 上方
+		core.drawImage(
+			ctx,
+			background,
+			128,
+			0,
+			16,
+			16,
+			x,
+			y,
+			16 * size,
+			16 * size
+		);
+		for (var dx = 0; dx < w - 64 * size; dx += 32 * size) {
+			core.drawImage(
+				ctx,
+				background,
+				144,
+				0,
+				32,
+				16,
+				x + dx + 16 * size,
+				y,
+				32 * size,
+				16 * size
+			);
+			core.drawImage(
+				ctx,
+				background,
+				144,
+				48,
+				32,
+				16,
+				x + dx + 16 * size,
+				y + h - 16 * size,
+				32 * size,
+				16 * size
+			);
+		}
+		core.drawImage(
+			ctx,
+			background,
+			144,
+			0,
+			(w - dx - 32 * size) / size,
+			16,
+			x + dx + 16 * size,
+			y,
+			w - dx - 32 * size,
+			16 * size
+		);
+		core.drawImage(
+			ctx,
+			background,
+			144,
+			48,
+			(w - dx - 32 * size) / size,
+			16,
+			x + dx + 16 * size,
+			y + h - 16 * size,
+			w - dx - 32 * size,
+			16 * size
+		);
+		core.drawImage(
+			ctx,
+			background,
+			176,
+			0,
+			16,
+			16,
+			x + w - 16 * size,
+			y,
+			16 * size,
+			16 * size
+		);
+		// 左右
+		for (var dy = 0; dy < h - 64 * size; dy += 32 * size) {
+			core.drawImage(
+				ctx,
+				background,
+				128,
+				16,
+				16,
+				32,
+				x,
+				y + dy + 16 * size,
+				16 * size,
+				32 * size
+			);
+			core.drawImage(
+				ctx,
+				background,
+				176,
+				16,
+				16,
+				32,
+				x + w - 16 * size,
+				y + dy + 16 * size,
+				16 * size,
+				32 * size
+			);
+		}
+		core.drawImage(
+			ctx,
+			background,
+			128,
+			16,
+			16,
+			(h - dy - 32 * size) / size,
+			x,
+			y + dy + 16 * size,
+			16 * size,
+			h - dy - 32 * size
+		);
+		core.drawImage(
+			ctx,
+			background,
+			176,
+			16,
+			16,
+			(h - dy - 32 * size) / size,
+			x + w - 16 * size,
+			y + dy + 16 * size,
+			16 * size,
+			h - dy - 32 * size
+		);
+		//下方
+		core.drawImage(
+			ctx,
+			background,
+			128,
+			48,
+			16,
+			16,
+			x,
+			y + h - 16 * size,
+			16 * size,
+			16 * size
+		);
+		core.drawImage(
+			ctx,
+			background,
+			176,
+			48,
+			16,
+			16,
+			x + w - 16 * size,
+			y + h - 16 * size,
+			16 * size,
+			16 * size
+		);
+
+		// arrow
+		if (px != null && py != null) {
+			if (direction == "up") {
+				core.drawImage(
+					ctx,
+					background,
+					128,
+					96,
+					32,
+					32,
+					px,
+					y + h - 3 * size,
+					32 * size,
+					32 * size
+				);
+			} else if (direction == "down") {
+				core.drawImage(
+					ctx,
+					background,
+					160,
+					96,
+					32,
+					32,
+					px,
+					y - 29 * size,
+					32 * size,
+					32 * size
+				);
+			}
+		}
+		// 仿RM窗口皮肤 ↑
+	};
+	events.prototype.battle = function (id, x, y, force, callback) {
+		core.saveAndStopAutomaticRoute();
+		id = id || core.getBlockId(x, y);
+		const cls = core.getClsFromId(id);
+		if (!id || !cls || !(cls === "enemys" || cls === "enemy48"))
+			return core.clearContinueAutomaticRoute(callback);
+		// 非强制战斗
+		if (core.enemys.getDamage(id, x, y) === null && !force && !core.status.event.id) {
+			core.stopSound();
+			core.playSound("操作失败");
+			core.drawTip("你打不过此怪物!", id);
+			return core.clearContinueAutomaticRoute(callback);
+		}
+		// 自动存档
+		if (!core.status.event.id) core.autosave(true);
+		// 战前事件
+		if (!this.beforeBattle(id, x, y))
+			return core.clearContinueAutomaticRoute(callback);
+		// 战后事件
+		this.afterBattle(id, x, y);
+		if (callback) callback();
+	};
+	actions.prototype._sys_ondown_lockControl = function (x, y, px, py) {
+		if (core.status.played && !core.status.lockControl) return false;
+
+		switch (core.status.event.id) {
+		case "battle":
+			core.plugin.battle_onclick(x, y, px, py);
+			break;
+		case "centerFly":
+			this._clickCenterFly(x, y, px, py);
+			break;
+		case "book":
+			this._clickBook(x, y, px, py);
+			break;
+		case "book-detail":
+			this._clickBookDetail(x, y, px, py);
+			break;
+		case "fly":
+			this._clickFly(x, y, px, py);
+			break;
+		case "viewMaps":
+			this._clickViewMaps(x, y, px, py);
+			break;
+		case "switchs":
+			this._clickSwitchs(x, y, px, py);
+			break;
+		case "switchs-sounds":
+			this._clickSwitchs_sounds(x, y, px, py);
+			break;
+		case "switchs-display":
+			this._clickSwitchs_display(x, y, px, py);
+			break;
+		case "switchs-action":
+			this._clickSwitchs_action(x, y, px, py);
+			break;
+		case "settings":
+			this._clickSettings(x, y, px, py);
+			break;
+		case "selectShop":
+			this._clickQuickShop(x, y, px, py);
+			break;
+		case "equipbox":
+			this._clickEquipbox(x, y, px, py);
+			break;
+		case "toolbox":
+			this._clickToolbox(x, y, px, py);
+			break;
+		case "save":
+		case "load":
+		case "replayLoad":
+		case "replayRemain":
+		case "replaySince":
+			this._clickSL(x, y, px, py);
+			break;
+		case "confirmBox":
+			this._clickConfirmBox(x, y, px, py);
+			break;
+		case "keyBoard":
+			this._clickKeyBoard(x, y, px, py);
+			break;
+		case "action":
+			this._clickAction(x, y, px, py);
+			break;
+		case "text":
+			core.drawText();
+			break;
+		case "notes":
+			this._clickNotes(x, y, px, py);
+			break;
+		case "syncSave":
+			this._clickSyncSave(x, y, px, py);
+			break;
+		case "syncSelect":
+			this._clickSyncSelect(x, y, px, py);
+			break;
+		case "localSaveSelect":
+			this._clickLocalSaveSelect(x, y, px, py);
+			break;
+		case "storageRemove":
+			this._clickStorageRemove(x, y, px, py);
+			break;
+		case "cursor":
+			this._clickCursor(x, y, px, py);
+			break;
+		case "replay":
+			this._clickReplay(x, y, px, py);
+			break;
+		case "gameInfo":
+			this._clickGameInfo(x, y, px, py);
+			break;
+		case "about":
+		case "help":
+			core.ui.closePanel();
+			break;
+		}
+
+		// --- 长按判定
+		if (core.timeout.onDownTimeout == null) {
+			core.timeout.onDownTimeout = setTimeout(function () {
+				if (core.interval.onDownInterval == null) {
+					core.interval.onDownInterval = setInterval(function () {
+						if (!core.actions.longClick(x, y, px, py)) {
+							clearInterval(core.interval.onDownInterval);
+							core.interval.onDownInterval = null;
+						}
+					}, 40);
+				}
+			}, 500);
+		}
+		return true;
+	};
+	core.registerAction(
+		"ondown",
+		"_sys_ondown_lockControl",
+		core.actions._sys_ondown_lockControl,
+		30
+	);
+	////// 结束一切事件和绘制,关闭UI窗口,返回游戏进程 //////
+	ui.prototype.closePanel = function () {
+		core.dom.saveLoad.style.display = "none"
+		if (core.status.hero && core.status.hero.flags) {
+			// 清除全部临时变量
+			Object.keys(core.status.hero.flags).forEach(function (name) {
+				if (name.startsWith("@temp@") || /^arg\d+$/.test(name)) {
+					delete core.status.hero.flags[name];
+				}
+			});
+		}
+		this.clearUI();
+		core.maps.generateGroundPattern();
+		core.updateStatusBar(true);
+		core.unlockControl();
+		core.status.event.data = null;
+		core.status.event.id = null;
+		core.status.event.selection = null;
+		core.status.event.ui = null;
+		core.status.event.interval = null;
+		// 清除onDownInterval
+		clearInterval(core.interval.onDownInterval);
+		core.interval.onDownInterval = 'tmp';
+	}
+	////// 绘制提示同时播放错误音效 //////
+	this.drawFailTip = function (text, id, frame) {
+		core.drawTip(text, id, frame);
+		core.playSound('error.opus');
+	}
+	const floor = Math.floor
+	Math.floor = (a) => floor(a + 1e-8)
+},
     "额外信息": function () {
     /* 宝石血瓶左下角显示数值
      * 注意!!!不要在道具属性中直接操作flags,使用core.status.hero.flags或core.setFlag系列函数代替!
@@ -4195,2143 +4746,2626 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
       }
     }
   },
+    "跳伤": function () {
+	// 在此增加新插件
+	/** 
+	函数使用说明
+	Dove.MorePerform.ShowDamagePop.PopDamage(
+    'ui', // 画布名称或画布2d上下文对象
+    100, 200, // 位置 (x, y)
+    -50, // 伤害值
+    24, // 字体大小
+	"宋体",//字体
+    '#FF0000', // 字体颜色
+    '#000000', // 描边颜色
+    0.5, // 水平速度 (speedX)
+    -10, // 垂直速度 (speedY)
+    0.5, // 重力 (gravity)
+    120 // 显示时长(帧数)
+	*/
+	if (!core.registerAnimationFrame) {
+		throw new Error("require 2.6.1 or higher version");
+	}
+
+	window.Dove = window.Dove || {};
+	Dove.MorePerform = Dove.MorePerform || {};
+
+	Dove.MorePerform.ShowDamagePop = {};
+	Dove.MorePerform.ShowDamagePop.version = 1.0;
+
+	Dove.MorePerform.ShowDamagePop.AllPopingCanvas = [];
+
+	// 每帧的处理
+	Dove.MorePerform.ShowDamagePop.Update = function () {
+		this.AllPopingCanvas = this.AllPopingCanvas.filter(function (spr) {
+			spr.update();
+			return spr.isAlive();
+		});
+		if (!this.AllPopingCanvas.length) PopSprite._count = 0;
+	};
+
+	// 弹出伤害气泡
+	Dove.MorePerform.ShowDamagePop.PopDamage = function (
+		canvasName,
+		x,
+		y,
+		damageValue,
+		fontSize,
+		font,
+		fontColor,
+		outlineColor,
+		speedX,
+		speedY,
+		gravity,
+		duration
+	) {
+		if (damageValue) {
+			var poper = new PopSprite(
+				canvasName,
+				x,
+				y,
+				damageValue,
+				fontSize,
+				font,
+				fontColor,
+				outlineColor,
+				speedX,
+				speedY,
+				gravity,
+				duration
+			);
+			Dove.MorePerform.ShowDamagePop.AllPopingCanvas.push(poper);
+		}
+	};
+
+	// 战斗发生前后记录生命值并处理
+	Dove.MorePerform.ShowDamagePop.OnBattle = core.events.battle;
+	core.events.battle = function () {
+		var hpBeforeBattle = core.status.hero.hp;
+		Dove.MorePerform.ShowDamagePop.OnBattle.apply(core.events, arguments);
+		if (core.getFlag("noAnimate"))
+			Dove.MorePerform.ShowDamagePop.PopDamage(
+				"ui", // 默认画布名称
+				core.getHeroLoc("x") * 32, // 英雄位置 x
+				core.getHeroLoc("y") * 32, // 英雄位置 y
+				Math.floor(core.status.hero.hp - hpBeforeBattle), // 伤害值
+				16, // 默认字体大小
+				"Arial", //默认字体
+				null, // 默认颜色
+				null, // 默认描边颜色
+				null, // 默认水平速度
+				null, // 默认垂直速度
+				null, // 默认重力
+				90 // 默认显示时长(帧数)
+			);
+	};
+	// 注册每帧事件
+	core.registerAnimationFrame("ShowDamagePop", true, (temptime) => {
+
+		Dove.MorePerform.ShowDamagePop.Update.bind(
+			Dove.MorePerform.ShowDamagePop
+		)();
+
+	});
+
+	// 弹出精灵类
+	function PopSprite(
+		canvasName,
+		x,
+		y,
+		damage,
+		fontSize,
+		font,
+		fontColor,
+		outlineColor,
+		speedX,
+		speedY,
+		gravity,
+		duration
+	) {
+		this.initialize.apply(this, arguments);
+	}
+
+	PopSprite.prototype = Object.create(Object.prototype);
+	PopSprite.prototype.constructor = PopSprite;
+
+	// 常量
+	PopSprite._count = 0;
+	PopSprite._baseZOrder = 50;
+	PopSprite._floorDis = 20;
+
+	// 初始化
+	PopSprite.prototype.initialize = function (
+		canvasName,
+		x,
+		y,
+		damage,
+		fontSize,
+		font,
+		fontColor,
+		outlineColor,
+		speedX,
+		speedY,
+		gravity,
+		duration
+	) {
+		this._canvasName = canvasName ?? "ui"; // 默认画布名称
+		this._x = x;
+		this._y = y;
+		this._damage = damage;
+		this._fontSize = fontSize ?? 16; // 默认字体大小
+		this._font = font ?? "Arial";
+		this._fontColor = fontColor ?? (damage > 0 ? "#22FF44" : "lightcoral"); // 默认颜色
+		this._outlineColor = outlineColor ?? "#FFFFFF"; // 默认描边颜色
+		this._speedX = speedX ?? -1 + Math.random() * 2; // 水平速度,默认随机
+		this._speedY = speedY ?? -3 - Math.random() * 4; // 垂直速度,默认随机
+		this._gravity = gravity ?? 0.3; // 重力加速度,默认 0.3
+		this._duration = duration ?? 180; // 显示时长(帧数),默认 180 帧
+		this.initAllMembers();
+		this.requestCanvas();
+	};
+
+	// 自更新
+	PopSprite.prototype.update = function () {
+		if (this._timer < this._duration) {
+			// 使用传入的显示时长
+			this._x += this._vx;
+			this._y += this._vy;
+			this._vy += this._gravity;
+			if (this._y >= this._floorY) {
+				this._y = this._floorY;
+				this._vy *= -0.75; // 反弹衰减
+			}
+			core.relocateCanvas(this._symbol, this._x, this._y);
+			core.setOpacity(this._symbol, 1 - this._timer / this._duration); // 根据显示时长设置透明度
+		} else {
+			this.dispose();
+		}
+		this._timer++;
+	};
+
+	// 申请并描绘canvas
+	PopSprite.prototype.requestCanvas = function () {
+		core.createCanvas(
+			this._symbol,
+			this._x,
+			this._y,
+			this._width + 4,
+			this._height + 4,
+			this._z
+		);
+
+		var canvas = core.getContextByName(this._symbol);
+		canvas.font = this._fontSize + "px " + this._font; // 动态设置字体大小
+		canvas.fillStyle = this._fontColor; // 动态设置字体颜色
+		canvas.strokeStyle = this._outlineColor; // 动态设置描边颜色
+		canvas.strokeText(this._text, 2, this._height);
+		canvas.fillText(this._text, 2, this._height);
+	};
+
+	// 初始化所有成员变量
+	PopSprite.prototype.initAllMembers = function () {
+		this._text = String(this._damage);
+		var uiContext = core.ui.getContextByName(this._canvasName); // 使用指定画布
+		uiContext.font = this._fontSize + "px  " + this._font; // 动态设置字体大小
+		var textRect = uiContext.measureText(this._text);
+		this._width = textRect.width + 4;
+		this._height = this._fontSize + 4; // 动态设置高度
+		this._z = uiContext.canvas.style.zIndex ?
+			Number(uiContext.canvas.style.zIndex) + PopSprite._count :
+			PopSprite._baseZOrder + PopSprite._count;
+		this._symbol = "popSprite" + PopSprite._count++;
+		this._alive = true;
+		this._vx = this._speedX; // 使用传入的水平速度
+		this._vy = this._speedY; // 使用传入的垂直速度
+		this._floorY = this._y + PopSprite._floorDis;
+		this._timer = 0;
+	};
+
+	// 判断是否存活
+	PopSprite.prototype.isAlive = function () {
+		return this._alive;
+	};
+
+	// 释放
+	PopSprite.prototype.dispose = function () {
+		this._alive = false;
+		core.deleteCanvas(this._symbol);
+	};
+},
     "编辑器显伤": function () {
-    // 在此增加新插件
-    /////// 用户设置 ///////
-    // 将__enable置为false将关闭插件
-    var __enable = true;
-    // 魔防攻速之类的属性可以在这里加 ['atk', 'def', 'mdef']
-    var heroStatus = ["atk", "def", "mdef", "hp"];
-    // saveHero为true 将会把每次造塔测试时的角色数据存下来 否则会读取初始属性
-    // 用不着可以关了 节约缓存空间 (虽然根本没多少 还没一个存档大
-    // 也可以手动清理 控制台输入core.removeLocalStorage('editorHero')即可
-    var saveHero = true;
+	// 在此增加新插件
+	/////// 用户设置 ///////
+	// 将__enable置为false将关闭插件
+	var __enable = true;
+	// 魔防攻速之类的属性可以在这里加 ['atk', 'def', 'mdef']
+	var heroStatus = [
+		"atk",
+		"def",
+		"spell",
+		"mdef",
+		"matk",
+		"mhp",
+		"speed",
+		"hp",
+	];
+	// saveHero为true 将会把每次造塔测试时的角色数据存下来 否则会读取初始属性
+	// 用不着可以关了 节约缓存空间 (虽然根本没多少 还没一个存档大
+	// 也可以手动清理 控制台输入core.removeLocalStorage('editorHero')即可
+	var saveHero = true;
 
-    // 下为具体实现 懒得写注释了 大概就是写HTML然后注册交互
-    if (!__enable || main.mode != "editor") return;
-    core.plugin.initEditorDamage = false;
-    if (heroStatus.length >= 4 && !editor.isMobile)
-      editor.dom.mid2.style.top = 650 + 30 * (heroStatus.length - 3) + "px";
-    editor.statusRatio = core.getLocalStorage("statusRatio", 1);
-    editor.saveHero = saveHero;
-    editor._heroStatus = heroStatus;
-    editor.dom.mapEdit.appendChild(core.canvas.damage.canvas);
-    var HTML =
-      "<input type='button' value='←'/><input type='button' value='↑'/><input type='button' value='↓'/><input type='button' value='→'/><input type='button' id='bigmapBtn' value='大地图'' style='margin-left: '5px'/>";
+	// 下为具体实现 懒得写注释了 大概就是写HTML然后注册交互
+	if (!__enable || main.mode != "editor") return;
+	core.plugin.initEditorDamage = false;
+	if (heroStatus.length >= 4 && !editor.isMobile) {
+		editor.dom.mid2.style.top = 650 + 30 * (heroStatus.length - 3) + "px";
+		editor.dom.mid.style.height = 730 + "px";
+		document.querySelector("#mid .tools").style.height = 280 + "px";
+	}
+	editor.statusRatio = core.getLocalStorage("statusRatio", 1);
+	editor.saveHero = saveHero;
+	editor._heroStatus = heroStatus;
+	editor.dom.mapEdit.appendChild(core.canvas.damage.canvas);
+	const viewportButtons = document.getElementById("viewportButtons");
 
-    //if (heroStatus.length >= 4 && !editor.isMobile) editor.dom.mid2.style.top = 650 + 30 * (heroStatus.length - 3) + 'px';
-    heroStatus.forEach(function (status) {
-      var id = status + "set",
-        id2 = status + "add",
-        id3 = status + "rec",
-        id4 = status + "help";
-      HTML +=
-        "<br/><input type='text' size='15' id='" +
-        id +
-        "'><input type='button' id='" +
-        id2 +
-        "' value = '+'><input type='button' id='" +
-        id3 +
-        "' value = '-'><input type='button' value='?' id = '" +
-        id4 +
-        "'>";
-    });
-    document.getElementById("viewportButtons").innerHTML = HTML;
-    ["set", "add", "rec", "help"].forEach(function (e) {
-      heroStatus.forEach(function (status) {
-        editor.dom[status + e] = document.getElementById(status + e);
-      });
-    });
-    var _hasItem = core.items.hasItem;
-    core.items.hasItem = function (itemId) {
-      if (itemId == "book" && main.mode == "editor") return true;
-      return _hasItem.call(core.items, itemId);
-    };
-    if (main.mode == "editor") {
-      var applyList = [
-        "getDamageString",
-        "nextCriticals",
-        "getEnemyInfo",
-        "getEnemyValue",
-      ];
-      applyList.forEach(function (name) {
-        var func = core.enemys[name];
-        core.enemys[name] = function () {
-          var args =
-            arguments.length === 1
-              ? [arguments[0]]
-              : Array.apply(null, arguments);
-          if (typeof args[0] == "string") args[0] = core.enemys.enemys[args[0]];
-          return func.apply(core.enemys, args);
-        };
-      });
-    }
+	var HTML =
+		"<input type='button' value='←'/><input type='button' value='↑'/><input type='button' value='↓'/><input type='button' value='→'/><input type='button' id='bigmapBtn' value='大地图'' style='margin-left: '5px'/>";
 
-    ////// 获得勇士属性 //////
-    core.control.getStatus = function (name) {
-      if (!core.status.hero) return null;
-      if (name == "x" || name == "y" || name == "direction")
-        return this.getHeroLoc(name);
-      /*if ( main.mode == 'editor' && !core.hasFlag('__statistics__')) {
+	//if (heroStatus.length >= 4 && !editor.isMobile) editor.dom.mid2.style.top = 650 + 30 * (heroStatus.length - 3) + 'px';
+	heroStatus.forEach(function (status) {
+		var id = status + "set",
+			id2 = status + "add",
+			id3 = status + "rec",
+			id4 = status + "help";
+		HTML +=
+			"<br/>" +
+			core.getStatusLabel(status) +
+			"<text ><input type='text' size='" +
+			(15 - core.getStatusLabel(status).length * 2.5) +
+			"' id='" +
+			id +
+			"'><input type='button' id='" +
+			id2 +
+			"' value = '+'><input type='button' id='" +
+			id3 +
+			"' value = '-'><input type='button' value='?' id = '" +
+			id4 +
+			"'>";
+	});
+	document.getElementById("viewportButtons").innerHTML = HTML;
+	["set", "add", "rec", "help"].forEach(function (e) {
+		heroStatus.forEach(function (status) {
+			editor.dom[status + e] = document.getElementById(status + e);
+		});
+	});
+	var _hasItem = core.items.hasItem;
+	core.items.hasItem = function (itemId) {
+		if (itemId == "book" && main.mode == "editor") return true;
+		return _hasItem.call(core.items, itemId);
+	};
+	if (main.mode == "editor") {
+		var applyList = [
+			"getDamageString",
+			"nextCriticals",
+			"getEnemyInfo",
+			"getEnemyValue",
+		];
+		applyList.forEach(function (name) {
+			var func = core.enemys[name];
+			core.enemys[name] = function () {
+				var args =
+					arguments.length === 1 ?
+					[arguments[0]] :
+					Array.apply(null, arguments);
+				if (typeof args[0] == "string") args[0] = core.enemys.enemys[args[0]];
+				return func.apply(core.enemys, args);
+			};
+		});
+	}
+
+	////// 获得勇士属性 //////
+	core.control.getStatus = function (name) {
+		if (!core.status.hero) return null;
+		if (name == "x" || name == "y" || name == "direction")
+			return this.getHeroLoc(name);
+		/*if ( main.mode == 'editor' && !core.hasFlag('__statistics__')) {
 			return data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.hero[name];
 		}*/
-      return core.status.hero[name];
-    };
+		return core.status.hero[name];
+	};
 
-    core.control.updateDamage = function (floorId, ctx) {
-      floorId = floorId || core.status.floorId;
-      if (!floorId || core.status.gameOver) return;
-      var onMap = ctx == null;
-      if (main.mode == "editor") {
-        ctx = core.canvas.damage;
-        core.updateCheckBlock();
-        core.clearMap(ctx);
-        if (editor.uivalues.bigmap) return;
-      }
+	core.control.updateDamage = function (floorId, ctx) {
+		floorId = floorId || core.status.floorId;
+		if (!floorId || core.status.gameOver) return;
+		var onMap = ctx == null;
+		if (main.mode == "editor") {
+			ctx = core.canvas.damage;
+			core.updateCheckBlock();
+			core.clearMap(ctx);
+			if (editor.uivalues.bigmap) return;
+		}
 
-      // 没有怪物手册
-      if (!core.hasItem("book")) return;
-      core.status.damage.posX = core.bigmap.posX;
-      core.status.damage.posY = core.bigmap.posY;
-      if (!onMap) {
-        var width = core.floors[floorId].width,
-          height = core.floors[floorId].height;
-        // 地图过大的缩略图不绘制显伤
-        if (width * height > core.bigmap.threshold) return;
-      }
-      this._updateDamage_damage(floorId, onMap);
-      this._updateDamage_extraDamage(floorId, onMap);
-      this.drawDamage(ctx);
-    };
+		// 没有怪物手册
+		if (!core.hasItem("book")) return;
+		core.status.damage.posX = core.bigmap.posX;
+		core.status.damage.posY = core.bigmap.posY;
+		if (!onMap) {
+			var width = core.floors[floorId].width,
+				height = core.floors[floorId].height;
+			// 地图过大的缩略图不绘制显伤
+			if (width * height > core.bigmap.threshold) return;
+		}
+		this._updateDamage_damage(floorId, onMap);
+		this._updateDamage_extraDamage(floorId, onMap);
+		this.drawDamage(ctx);
+	};
 
-    core.control.drawDamage = function (ctx) {
-      if (
-        core.status.gameOver ||
-        !core.status.damage /* || main.mode != 'play'*/
-      )
-        return;
-      var onMap = false;
-      if (ctx == null) {
-        ctx = core.canvas.damage;
-        core.clearMap("damage");
-        onMap = true;
-      }
+	core.control.drawDamage = function (ctx) {
+		if (
+			core.status.gameOver ||
+			!core.status.damage /* || main.mode != 'play'*/
+		)
+			return;
+		var onMap = false;
+		if (ctx == null) {
+			ctx = core.canvas.damage;
+			core.clearMap("damage");
+			onMap = true;
+		}
 
-      if (onMap && core.bigmap.v2) {
-        // 检查是否需要重算...
-        if (
-          Math.abs(core.bigmap.posX - core.status.damage.posX) >=
-            core.bigmap.extend - 1 ||
-          Math.abs(core.bigmap.posY - core.status.damage.posY) >=
-            core.bigmap.extend - 1
-        ) {
-          return this.updateDamage();
-        }
-      }
-      return this._drawDamage_draw(ctx, onMap);
-    };
+		if (onMap && core.bigmap.v2) {
+			// 检查是否需要重算...
+			if (
+				Math.abs(core.bigmap.posX - core.status.damage.posX) >=
+				core.bigmap.extend - 1 ||
+				Math.abs(core.bigmap.posY - core.status.damage.posY) >=
+				core.bigmap.extend - 1
+			) {
+				return this.updateDamage();
+			}
+		}
+		return this._drawDamage_draw(ctx, onMap);
+	};
 
-    ////// 以x,y的形式返回每个点的事件 //////
-    core.maps.getMapBlocksObj = function (floorId, noCache) {
-      floorId = floorId || core.status.floorId;
-      if (
-        core.status.mapBlockObjs[floorId] &&
-        !noCache &&
-        main.mode != "editor"
-      )
-        return core.status.mapBlockObjs[floorId];
+	////// 以x,y的形式返回每个点的事件 //////
+	core.maps.getMapBlocksObj = function (floorId, noCache) {
+		floorId = floorId || core.status.floorId;
+		if (
+			core.status.mapBlockObjs[floorId] &&
+			!noCache &&
+			main.mode != "editor"
+		)
+			return core.status.mapBlockObjs[floorId];
 
-      var obj = {};
-      core.extractBlocks(floorId);
-      core.status.maps[floorId].blocks.forEach(function (block) {
-        obj[block.x + "," + block.y] = block;
-      });
-      core.status.mapBlockObjs[floorId] = obj;
-      return obj;
-    };
+		var obj = {};
+		core.extractBlocks(floorId);
+		core.status.maps[floorId].blocks.forEach(function (block) {
+			obj[block.x + "," + block.y] = block;
+		});
+		core.status.mapBlockObjs[floorId] = obj;
+		return obj;
+	};
 
-    this.bignum = function (num, defaultValue) {
-      if (num == null || num == "") return defaultValue;
-      num = num + "";
-      var list = {
-        w: 1e4,
-        e: 1e8,
-        z: 1e12,
-        j: 1e16,
-        g: 1e20,
-      };
-      // 浮点数问题
-      function checkFloat(num) {
-        if (!core.isset(num)) return 0;
-        num = num + "";
-        var index = num.indexOf(".");
-        if (index < 0) return 0;
-        else return num.slice(index + 1).length;
-      }
-      var index = num.search(/w|e|z|j|g/);
-      if (index <= 0) {
-        num = parseInt(num);
-        if (core.isset(num)) return num;
-        else {
-          alert("不正确的输入");
-          return defaultValue;
-        }
-      }
-      for (; index > 0; index = num.search(/w|e|z|j|g/)) {
-        var p = num[index],
-          q = list[p],
-          n = num.slice(0, index),
-          m = Math.pow(10, checkFloat(n));
-        num = (n * m * q) / m + num.slice(index + 1);
-      }
-      return parseInt(num);
-    };
+	this.bignum = function (num, defaultValue) {
+		if (num == null || num == "") return defaultValue;
+		num = num + "";
+		var list = {
+			w: 1e4,
+			e: 1e8,
+			z: 1e12,
+			j: 1e16,
+			g: 1e20,
+		};
+		// 浮点数问题
+		function checkFloat(num) {
+			if (!core.isset(num)) return 0;
+			num = num + "";
+			var index = num.indexOf(".");
+			if (index < 0) return 0;
+			else return num.slice(index + 1).length;
+		}
+		var index = num.search(/w|e|z|j|g/);
+		if (index <= 0) {
+			num = parseInt(num);
+			if (core.isset(num)) return num;
+			else {
+				alert("不正确的输入");
+				return defaultValue;
+			}
+		}
+		for (; index > 0; index = num.search(/w|e|z|j|g/)) {
+			var p = num[index],
+				q = list[p],
+				n = num.slice(0, index),
+				m = Math.pow(10, checkFloat(n));
+			num = (n * m * q) / m + num.slice(index + 1);
+		}
+		return parseInt(num);
+	};
 
-    this.updateEditorDamage = function (noSave) {
-      core.updateDamage();
-      heroStatus.forEach(function (status) {
-        editor.dom[status + "set"].value = core.status.hero[status];
-      });
-      if (!noSave && editor.saveHero)
-        core.setLocalStorage("editorHero", core.status.hero);
-    };
+	this.updateEditorDamage = function (noSave) {
+		core.updateDamage();
+		heroStatus.forEach(function (status) {
+			editor.dom[status + "set"].value = core.status.hero[status];
+		});
+		if (!noSave && editor.saveHero)
+			core.setLocalStorage("editorHero", core.status.hero);
+	};
 
-    var _resizeMap = core.maps.resizeMap;
-    core.maps.resizeMap = function (floorId) {
-      _resizeMap.call(core.maps, floorId);
-      if (!core.plugin.initEditorDamage && main.mode == "editor") {
-        core.plugin.initEditorDamage = true;
-        var editorHero = core.getLocalStorage("editorHero");
-        if (editorHero && saveHero) core.status.hero = editorHero;
-        else core.removeLocalStorage("editorHero");
-        editor._heroStatus.forEach(function (e) {
-          editor.dom[e + "set"].onchange = function () {
-            var status = this.id.slice(0, -3);
-            core.status.hero[status] = core.bignum(
-              this.value,
-              core.status.hero[status]
-            );
-            core.updateEditorDamage();
-          };
-          editor.dom[e + "add"].onclick = function () {
-            var status = this.id.slice(0, -3);
-            core.status.hero[status] += editor.statusRatio;
-            core.updateEditorDamage();
-          };
-          editor.dom[e + "rec"].onclick = function () {
-            var status = this.id.slice(0, -3);
-            core.status.hero[status] -= editor.statusRatio;
-            core.updateEditorDamage();
-          };
-          editor.dom[e + "help"].onclick = function () {
-            var status = this.id.slice(0, -4),
-              name = core.getStatusLabel(status);
-            var ratio = parseInt(
-              prompt(
-                "当前属性:" +
-                  name +
-                  "\n现在的点击按钮变化值:" +
-                  editor.statusRatio +
-                  ",请输入按下一次+/-按钮的属性变化量,可以写4w 10.2e这种字母缩写"
-              )
-            );
-            if (!core.isset(ratio)) {
-              printe("不合法的输入");
-              return;
-            }
-            editor.statusRatio = ratio;
-            core.setLocalStorage("statusRatio", ratio);
-          };
-        });
-        var _updateMap = editor.updateMap;
-        editor.updateMap = function () {
-          _updateMap.call(editor);
-          core.updateEditorDamage(true);
-        };
-        editor.mode.onmode = function (mode, callback) {
-          if (editor_mode.mode != mode) {
-            if (mode === "save") {
-              editor_mode.doActionList(
-                editor_mode.mode,
-                editor_mode.actionList,
-                function () {
-                  if (callback) callback();
-                  core.updateEditorDamage();
-                }
-              );
-            }
-            if (editor_mode.mode === "nextChange" && mode)
-              editor_mode.showMode(mode);
-            if (mode !== "save") editor_mode.mode = mode;
-            editor_mode.actionList = [];
-          }
-        };
-      }
-    };
-  },
+	var _resizeMap = core.maps.resizeMap;
+	core.maps.resizeMap = function (floorId) {
+		_resizeMap.call(core.maps, floorId);
+		if (!core.plugin.initEditorDamage && main.mode == "editor") {
+			core.plugin.initEditorDamage = true;
+			var editorHero = core.getLocalStorage("editorHero");
+			if (editorHero && saveHero) core.status.hero = editorHero;
+			else core.removeLocalStorage("editorHero");
+			editor._heroStatus.forEach(function (e) {
+				editor.dom[e + "set"].onchange = function () {
+					var status = this.id.slice(0, -3);
+					core.status.hero[status] = core.bignum(
+						this.value,
+						core.status.hero[status]
+					);
+					if (status === "mdef" && core.status.hero[status] > 100)
+						core.status.hero[status] = 100;
+					core.updateEditorDamage();
+				};
+				editor.dom[e + "add"].onclick = function () {
+					var status = this.id.slice(0, -3);
+					core.status.hero[status] += editor.statusRatio;
+					if (status === "mdef" && core.status.hero[status] > 100)
+						core.status.hero[status] = 100;
+					core.updateEditorDamage();
+				};
+				editor.dom[e + "rec"].onclick = function () {
+					var status = this.id.slice(0, -3);
+					core.status.hero[status] -= editor.statusRatio;
+					core.updateEditorDamage();
+				};
+				editor.dom[e + "help"].onclick = function () {
+					var status = this.id.slice(0, -4),
+						name = core.getStatusLabel(status);
+					var ratio = parseInt(
+						prompt(
+							"当前属性:" +
+							name +
+							"\n现在的点击按钮变化值:" +
+							editor.statusRatio +
+							",请输入按下一次+/-按钮的属性变化量,可以写4w 10.2e这种字母缩写"
+						)
+					);
+					if (!core.isset(ratio)) {
+						printe("不合法的输入");
+						return;
+					}
+					editor.statusRatio = ratio;
+					core.setLocalStorage("statusRatio", ratio);
+				};
+			});
+			var _updateMap = editor.updateMap;
+			editor.updateMap = function () {
+				_updateMap.call(editor);
+				core.updateEditorDamage(true);
+			};
+			editor.mode.onmode = function (mode, callback) {
+				if (editor_mode.mode != mode) {
+					if (mode === "save") {
+						editor_mode.doActionList(
+							editor_mode.mode,
+							editor_mode.actionList,
+							function () {
+								if (callback) callback();
+								core.updateEditorDamage();
+							}
+						);
+					}
+					if (editor_mode.mode === "nextChange" && mode)
+						editor_mode.showMode(mode);
+					if (mode !== "save") editor_mode.mode = mode;
+					editor_mode.actionList = [];
+				}
+			};
+		}
+	};
+},
     "手册区分特殊属性": function () {
-    // 在此增加新插件
-    this.arrsame = function (Arraya, Arrayb) {
-      let a = Arraya || [];
-      let b = Arrayb || [];
-      if (typeof a === "number") a = [a];
-      if (typeof b === "number") b = [b];
-      let c = [...a, ...b];
-      for (const i of c)
-        if (!a.includes(i) || !b.includes(i)) {
-          return false;
-        }
-      return true;
-    };
-    enemys.prototype.getCurrentEnemys = function (floorId) {
-      floorId = floorId || core.status.floorId;
-      var enemys = [],
-        used = {};
-      core.extractBlocks(floorId);
-      core.status.maps[floorId].blocks.forEach(function (block) {
-        if (!block.disable && block.event.cls.indexOf("enemy") == 0) {
-          this._getCurrentEnemys_addEnemy(
-            block.event.id,
-            enemys,
-            used,
-            block.x,
-            block.y,
-            floorId
-          );
-        }
-      }, this);
-      return this._getCurrentEnemys_sort(enemys);
-    };
+	// 在此增加新插件
+	this.arrsame = function (Arraya, Arrayb) {
+		let a = Arraya || [];
+		let b = Arrayb || [];
+		if (typeof a === "number") a = [a];
+		if (typeof b === "number") b = [b];
+		let c = [...a, ...b];
+		for (const i of c)
+			if (!a.includes(i) || !b.includes(i)) {
+				return false;
+			}
+		return true;
+	};
+	enemys.prototype.getCurrentEnemys = function (floorId) {
+		floorId = floorId || core.status.floorId;
+		var enemys = [],
+			used = {};
+		core.extractBlocks(floorId);
+		core.status.maps[floorId].blocks.forEach(function (block) {
+			if (!block.disable && block.event.cls.indexOf("enemy") == 0) {
+				this._getCurrentEnemys_addEnemy(
+					block.event.id,
+					enemys,
+					used,
+					block.x,
+					block.y,
+					floorId
+				);
+			}
+		}, this);
+		return this._getCurrentEnemys_sort(enemys);
+	};
 
-    enemys.prototype._getCurrentEnemys_getEnemy = function (enemyId) {
-      var enemy = core.material.enemys[enemyId];
-      if (!enemy) return null;
+	enemys.prototype._getCurrentEnemys_getEnemy = function (enemyId) {
+		var enemy = core.material.enemys[enemyId];
+		if (!enemy) return null;
 
-      // 检查朝向;displayIdInBook
-      return (
-        core.material.enemys[enemy.displayIdInBook] ||
-        core.material.enemys[(enemy.faceIds || {}).down] ||
-        enemy
-      );
-    };
+		// 检查朝向;displayIdInBook
+		return (
+			core.material.enemys[enemy.displayIdInBook] ||
+			core.material.enemys[(enemy.faceIds || {}).down] ||
+			enemy
+		);
+	};
 
-    enemys.prototype._getCurrentEnemys_addEnemy = function (
-      enemyId,
-      enemys,
-      used,
-      x,
-      y,
-      floorId
-    ) {
-      var enemy = this._getCurrentEnemys_getEnemy(enemyId);
-      if (enemy == null) return;
+	enemys.prototype._getCurrentEnemys_addEnemy = function (
+		enemyId,
+		enemys,
+		used,
+		x,
+		y,
+		floorId
+	) {
+		var enemy = this._getCurrentEnemys_getEnemy(enemyId);
+		if (enemy == null) return;
 
-      var id = enemy.id;
+		var id = enemy.id;
 
-      var enemyInfo = this.getEnemyInfo(enemy, null, null, null, floorId);
-      var locEnemyInfo = this.getEnemyInfo(enemy, null, x, y, floorId);
-      if (
-        !core.flags.enableEnemyPoint ||
-        (locEnemyInfo.atk == enemyInfo.atk &&
-          locEnemyInfo.def == enemyInfo.def &&
-          locEnemyInfo.hp == enemyInfo.hp &&
-          core.plugin.arrsame(locEnemyInfo.special, enemyInfo.special))
-      ) {
-        x = null;
-        y = null;
-      } else {
-        // 检查enemys里面是否使用了存在的内容
-        for (var i = 0; i < enemys.length; ++i) {
-          var one = enemys[i];
-          if (
-            id == one.id &&
-            one.locs != null &&
-            locEnemyInfo.atk == one.atk &&
-            locEnemyInfo.def == one.def &&
-            locEnemyInfo.hp == one.hp &&
-            core.plugin.arrsame(locEnemyInfo.special, one.special)
-          ) {
-            one.locs.push([x, y]);
-            return;
-          }
-        }
-        enemyInfo = locEnemyInfo;
-      }
+		var enemyInfo = this.getEnemyInfo(enemy, null, null, null, floorId);
+		var locEnemyInfo = this.getEnemyInfo(enemy, null, x, y, floorId);
+		if (
+			!core.flags.enableEnemyPoint ||
+			(locEnemyInfo.atk == enemyInfo.atk &&
+				locEnemyInfo.def == enemyInfo.def &&
+				locEnemyInfo.hp == enemyInfo.hp &&
+				core.plugin.arrsame(locEnemyInfo.special, enemyInfo.special))
+		) {
+			x = null;
+			y = null;
+		} else {
+			// 检查enemys里面是否使用了存在的内容
+			for (var i = 0; i < enemys.length; ++i) {
+				var one = enemys[i];
+				if (
+					id == one.id &&
+					one.locs != null &&
+					locEnemyInfo.atk == one.atk &&
+					locEnemyInfo.def == one.def &&
+					locEnemyInfo.hp == one.hp &&
+					core.plugin.arrsame(locEnemyInfo.special, one.special)
+				) {
+					one.locs.push([x, y]);
+					return;
+				}
+			}
+			enemyInfo = locEnemyInfo;
+		}
 
-      var id = enemy.id + ":" + x + ":" + y;
-      if (used[id]) return;
-      used[id] = true;
+		var id = enemy.id + ":" + x + ":" + y;
+		if (used[id]) return;
+		used[id] = true;
 
-      var specialText = core.enemys.getSpecialText(enemy, x, y, floorId);
-      var specialColor = core.enemys.getSpecialColor(enemy, x, y, floorId);
+		var specialText = core.enemys.getSpecialText(enemy, x, y, floorId);
+		var specialColor = core.enemys.getSpecialColor(enemy, x, y, floorId);
 
-      var critical = this.nextCriticals(enemy, 1, x, y, floorId);
-      if (critical.length > 0) critical = critical[0];
+		var critical = this.nextCriticals(enemy, 1, x, y, floorId);
+		if (critical.length > 0) critical = critical[0];
+		var criticalSpeed = core.nextCriticals_speed(enemy, 1, x, y, floorId);
+		if (criticalSpeed.length > 0) criticalSpeed = criticalSpeed[0];
+		var criticalSpell = core.nextCriticals_spell(enemy, 1, x, y, floorId);
+		if (criticalSpell.length > 0) criticalSpell = criticalSpell[0];
+		var e = core.clone(enemy);
+		for (var v in enemyInfo) {
+			e[v] = enemyInfo[v];
+		}
+		if (x != null && y != null) {
+			e.locs = [
+				[x, y]
+			];
+		}
+		e.name = core.getEnemyValue(enemy, "name", x, y, floorId);
+		e.specialText = specialText;
+		e.specialColor = specialColor;
+		e.damage = this.getDamage(enemy, x, y, floorId);
+		e.critical = critical[0];
+		e.criticalDamage = critical[1];
+		e.criticalAtk = [critical[0], critical[1]];
+		e.criticalSpeed = criticalSpeed;
+		e.criticalSpell = criticalSpell;
+		e.defDamage = this._getCurrentEnemys_addEnemy_defDamage(
+			enemy,
+			x,
+			y,
+			floorId
+		);
+		enemys.push(e);
+	};
 
-      var e = core.clone(enemy);
-      for (var v in enemyInfo) {
-        e[v] = enemyInfo[v];
-      }
-      if (x != null && y != null) {
-        e.locs = [[x, y]];
-      }
-      e.name = core.getEnemyValue(enemy, "name", x, y, floorId);
-      e.specialText = specialText;
-      e.specialColor = specialColor;
-      e.damage = this.getDamage(enemy, x, y, floorId);
-      e.critical = critical[0];
-      e.criticalDamage = critical[1];
-      e.defDamage = this._getCurrentEnemys_addEnemy_defDamage(
-        enemy,
-        x,
-        y,
-        floorId
-      );
-      enemys.push(e);
-    };
+	enemys.prototype._getCurrentEnemys_addEnemy_defDamage = function (
+		enemy,
+		x,
+		y,
+		floorId
+	) {
+		var ratio = core.status.maps[floorId || core.status.floorId].ratio || 1;
+		return this.getDefDamage(enemy, ratio, x, y, floorId);
+	};
 
-    enemys.prototype._getCurrentEnemys_addEnemy_defDamage = function (
-      enemy,
-      x,
-      y,
-      floorId
-    ) {
-      var ratio = core.status.maps[floorId || core.status.floorId].ratio || 1;
-      return this.getDefDamage(enemy, ratio, x, y, floorId);
-    };
+	enemys.prototype._getCurrentEnemys_sort = function (enemys) {
+		return enemys.sort(function (a, b) {
+			if (a.damage == b.damage) {
+				return a.money - b.money;
+			}
+			if (a.damage == null) {
+				return 1;
+			}
+			if (b.damage == null) {
+				return -1;
+			}
+			return a.damage - b.damage;
+		});
+	};
 
-    enemys.prototype._getCurrentEnemys_sort = function (enemys) {
-      return enemys.sort(function (a, b) {
-        if (a.damage == b.damage) {
-          return a.money - b.money;
-        }
-        if (a.damage == null) {
-          return 1;
-        }
-        if (b.damage == null) {
-          return -1;
-        }
-        return a.damage - b.damage;
-      });
-    };
+	////// 获得所有特殊属性的名称 //////
+	enemys.prototype.getSpecialText = function (enemy, x, y, floorId) {
+		if (typeof enemy == "string")
+			enemy = this.getEnemyInfo(enemy, null, x, y, floorId);
+		if (!enemy) return [];
+		var special = enemy.special;
+		var text = [];
 
-    ////// 获得所有特殊属性的名称 //////
-    enemys.prototype.getSpecialText = function (enemy, x, y, floorId) {
-      if (typeof enemy == "string")
-        enemy = this.getEnemyInfo(enemy, null, x, y, floorId);
-      if (!enemy) return [];
-      var special = enemy.special;
-      var text = [];
+		var specials = this.getSpecials();
+		if (specials) {
+			for (var i = 0; i < specials.length; i++) {
+				if (this.hasSpecial(special, specials[i][0]))
+					text.push(this._calSpecialContent(enemy, specials[i][1]));
+			}
+		}
+		return text;
+	};
 
-      var specials = this.getSpecials();
-      if (specials) {
-        for (var i = 0; i < specials.length; i++) {
-          if (this.hasSpecial(special, specials[i][0]))
-            text.push(this._calSpecialContent(enemy, specials[i][1]));
-        }
-      }
-      return text;
-    };
+	////// 获得所有特殊属性的颜色 //////
+	enemys.prototype.getSpecialColor = function (enemy, x, y, floorId) {
+		if (typeof enemy == "string")
+			enemy = this.getEnemyInfo(enemy, null, x, y, floorId);
+		if (!enemy) return [];
+		var special = enemy.special;
+		var colors = [];
 
-    ////// 获得所有特殊属性的颜色 //////
-    enemys.prototype.getSpecialColor = function (enemy, x, y, floorId) {
-      if (typeof enemy == "string")
-        enemy = this.getEnemyInfo(enemy, null, x, y, floorId);
-      if (!enemy) return [];
-      var special = enemy.special;
-      var colors = [];
+		var specials = this.getSpecials();
+		if (specials) {
+			for (var i = 0; i < specials.length; i++) {
+				if (this.hasSpecial(special, specials[i][0]))
+					colors.push(specials[i][3] || null);
+			}
+		}
+		return colors;
+	};
 
-      var specials = this.getSpecials();
-      if (specials) {
-        for (var i = 0; i < specials.length; i++) {
-          if (this.hasSpecial(special, specials[i][0]))
-            colors.push(specials[i][3] || null);
-        }
-      }
-      return colors;
-    };
+	////// 获得所有特殊属性的额外标记 //////
+	enemys.prototype.getSpecialFlag = function (enemy, x, y, floorId) {
+		if (typeof enemy == "string")
+			enemy = getEnemyInfo(enemy, null, x, y, floorId);
+		if (!enemy) return [];
+		var special = enemy.special;
+		var flag = 0;
 
-    ////// 获得所有特殊属性的额外标记 //////
-    enemys.prototype.getSpecialFlag = function (enemy, x, y, floorId) {
-      if (typeof enemy == "string")
-        enemy = getEnemyInfo(enemy, null, x, y, floorId);
-      if (!enemy) return [];
-      var special = enemy.special;
-      var flag = 0;
+		var specials = this.getSpecials();
+		if (specials) {
+			for (var i = 0; i < specials.length; i++) {
+				if (this.hasSpecial(special, specials[i][0]))
+					flag |= specials[i][4] || 0;
+			}
+		}
+		return flag;
+	};
 
-      var specials = this.getSpecials();
-      if (specials) {
-        for (var i = 0; i < specials.length; i++) {
-          if (this.hasSpecial(special, specials[i][0]))
-            flag |= specials[i][4] || 0;
-        }
-      }
-      return flag;
-    };
+	////// 获得每个特殊属性的说明 //////
+	enemys.prototype.getSpecialHint = function (enemy, special) {
+		var specials = this.getSpecials();
+		if (special == null) {
+			if (specials == null) return [];
+			var hints = [];
+			for (var i = 0; i < specials.length; i++) {
+				if (this.hasSpecial(enemy, specials[i][0]))
+					hints.push(
+						"\r[" +
+						core.arrayToRGBA(specials[i][3] || "#FF6A6A") +
+						"]\\d" +
+						this._calSpecialContent(enemy, specials[i][1]) +
+						":\\d\r[]" +
+						this._calSpecialContent(enemy, specials[i][2])
+					);
+			}
+			return hints;
+		}
 
-    ////// 获得每个特殊属性的说明 //////
-    enemys.prototype.getSpecialHint = function (enemy, special) {
-      var specials = this.getSpecials();
-      if (special == null) {
-        if (specials == null) return [];
-        var hints = [];
-        for (var i = 0; i < specials.length; i++) {
-          if (this.hasSpecial(enemy, specials[i][0]))
-            hints.push(
-              "\r[" +
-                core.arrayToRGBA(specials[i][3] || "#FF6A6A") +
-                "]\\d" +
-                this._calSpecialContent(enemy, specials[i][1]) +
-                ":\\d\r[]" +
-                this._calSpecialContent(enemy, specials[i][2])
-            );
-        }
-        return hints;
-      }
-
-      if (specials == null) return "";
-      for (var i = 0; i < specials.length; i++) {
-        if (special == specials[i][0])
-          return (
-            "\r[#FF6A6A]\\d" +
-            this._calSpecialContent(enemy, specials[i][1]) +
-            ":\\d\r[]" +
-            this._calSpecialContent(enemy, specials[i][2])
-          );
-      }
-      return "";
-    };
-    ui.prototype._drawBook_drawName = function (
-      index,
-      enemy,
-      top,
-      left,
-      width
-    ) {
-      // 绘制第零列(名称和特殊属性)
-      // 如果需要添加自己的比如怪物的称号等,也可以在这里绘制
-      core.setTextAlign("ui", "center");
-      if (core.enemys.getSpecialText(enemy).length == 0) {
-        core.fillText(
-          "ui",
-          enemy.name,
-          left + width / 2,
-          top + 35,
-          "#DDDDDD",
-          this._buildFont(17, true),
-          width
-        );
-      } else {
-        core.fillText(
-          "ui",
-          enemy.name,
-          left + width / 2,
-          top + 28,
-          "#DDDDDD",
-          this._buildFont(17, true),
-          width
-        );
-        switch (core.enemys.getSpecialText(enemy).length) {
-          case 1:
-            core.fillText(
-              "ui",
-              core.enemys.getSpecialText(enemy)[0],
-              left + width / 2,
-              top + 50,
-              core.arrayToRGBA(
-                (core.enemys.getSpecialColor(enemy) || [])[0] || "#FF6A6A"
-              ),
-              this._buildFont(15, true),
-              width
-            );
-            break;
-          case 2:
-            // Step 1: 计算字体
-            var text =
-              core.enemys.getSpecialText(enemy)[0] +
-              "  " +
-              core.enemys.getSpecialText(enemy)[1];
-            core.setFontForMaxWidth(
-              "ui",
-              text,
-              width,
-              this._buildFont(15, true)
-            );
-            // Step 2: 计算总宽度
-            var totalWidth = core.calWidth("ui", text);
-            var leftWidth = core.calWidth(
-              "ui",
-              core.enemys.getSpecialText(enemy)[0]
-            );
-            var rightWidth = core.calWidth(
-              "ui",
-              core.enemys.getSpecialText(enemy)[1]
-            );
-            // Step 3: 绘制
-            core.fillText(
-              "ui",
-              core.enemys.getSpecialText(enemy)[0],
-              left + (width + leftWidth - totalWidth) / 2,
-              top + 50,
-              core.arrayToRGBA(
-                (core.enemys.getSpecialColor(enemy) || [])[0] || "#FF6A6A"
-              )
-            );
-            core.fillText(
-              "ui",
-              core.enemys.getSpecialText(enemy)[1],
-              left + (width + totalWidth - rightWidth) / 2,
-              top + 50,
-              core.arrayToRGBA(
-                (core.enemys.getSpecialColor(enemy) || [])[1] || "#FF6A6A"
-              )
-            );
-            break;
-          default:
-            core.fillText(
-              "ui",
-              "多属性...",
-              left + width / 2,
-              top + 50,
-              "#FF6A6A",
-              this._buildFont(15, true),
-              width
-            );
-        }
-      }
-    };
-    ui.prototype._drawBookDetail_getInfo = function (index) {
-      var floorId =
-        core.floorIds[(core.status.event.ui || {}).index] ||
-        core.status.floorId;
-      // 清除浏览地图时的光环缓存
-      if (floorId != core.status.floorId && core.status.checkBlock) {
-        core.status.checkBlock.cache = {};
-      }
-      var enemys = core.enemys.getCurrentEnemys(floorId);
-      console.log(123);
-      if (enemys.length == 0) return [];
-      index = core.clamp(index, 0, enemys.length - 1);
-      var enemy = enemys[index];
-      var texts = core.enemys.getSpecialHint(enemy);
-      if (texts.length == 0) texts.push("该怪物无特殊属性。");
-      if (enemy.description) texts.push(enemy.description + "\r");
-      this._drawBookDetail_getTexts(enemy, floorId, texts);
-      texts.push("");
-      return [enemy, texts];
-    };
-  },
+		if (specials == null) return "";
+		for (var i = 0; i < specials.length; i++) {
+			if (special == specials[i][0])
+				return (
+					"\r[#FF6A6A]\\d" +
+					this._calSpecialContent(enemy, specials[i][1]) +
+					":\\d\r[]" +
+					this._calSpecialContent(enemy, specials[i][2])
+				);
+		}
+		return "";
+	};
+	ui.prototype._drawBook_drawName = function (
+		index,
+		enemy,
+		top,
+		left,
+		width
+	) {
+		// 绘制第零列(名称和特殊属性)
+		// 如果需要添加自己的比如怪物的称号等,也可以在这里绘制
+		core.setTextAlign("ui", "center");
+		if (core.enemys.getSpecialText(enemy).length == 0) {
+			core.fillText(
+				"ui",
+				enemy.name,
+				left + width / 2,
+				top + 35,
+				"#DDDDDD",
+				this._buildFont(17, true),
+				width
+			);
+		} else {
+			core.fillText(
+				"ui",
+				enemy.name,
+				left + width / 2,
+				top + 28,
+				"#DDDDDD",
+				this._buildFont(17, true),
+				width
+			);
+			switch (core.enemys.getSpecialText(enemy).length) {
+			case 1:
+				core.fillText(
+					"ui",
+					core.enemys.getSpecialText(enemy)[0],
+					left + width / 2,
+					top + 50,
+					core.arrayToRGBA(
+						(core.enemys.getSpecialColor(enemy) || [])[0] || "#FF6A6A"
+					),
+					this._buildFont(15, true),
+					width
+				);
+				break;
+			case 2:
+				// Step 1: 计算字体
+				var text =
+					core.enemys.getSpecialText(enemy)[0] +
+					"  " +
+					core.enemys.getSpecialText(enemy)[1];
+				core.setFontForMaxWidth(
+					"ui",
+					text,
+					width,
+					this._buildFont(15, true)
+				);
+				// Step 2: 计算总宽度
+				var totalWidth = core.calWidth("ui", text);
+				var leftWidth = core.calWidth(
+					"ui",
+					core.enemys.getSpecialText(enemy)[0]
+				);
+				var rightWidth = core.calWidth(
+					"ui",
+					core.enemys.getSpecialText(enemy)[1]
+				);
+				// Step 3: 绘制
+				core.fillText(
+					"ui",
+					core.enemys.getSpecialText(enemy)[0],
+					left + (width + leftWidth - totalWidth) / 2,
+					top + 50,
+					core.arrayToRGBA(
+						(core.enemys.getSpecialColor(enemy) || [])[0] || "#FF6A6A"
+					)
+				);
+				core.fillText(
+					"ui",
+					core.enemys.getSpecialText(enemy)[1],
+					left + (width + totalWidth - rightWidth) / 2,
+					top + 50,
+					core.arrayToRGBA(
+						(core.enemys.getSpecialColor(enemy) || [])[1] || "#FF6A6A"
+					)
+				);
+				break;
+			default:
+				core.fillText(
+					"ui",
+					"多属性...",
+					left + width / 2,
+					top + 50,
+					"#FF6A6A",
+					this._buildFont(15, true),
+					width
+				);
+			}
+		}
+	};
+	ui.prototype._drawBookDetail_getInfo = function (index) {
+		var floorId =
+			core.floorIds[(core.status.event.ui || {}).index] ||
+			core.status.floorId;
+		// 清除浏览地图时的光环缓存
+		if (floorId != core.status.floorId && core.status.checkBlock) {
+			core.status.checkBlock.cache = {};
+		}
+		var enemys = core.enemys.getCurrentEnemys(floorId);
+		console.log(123);
+		if (enemys.length == 0) return [];
+		index = core.clamp(index, 0, enemys.length - 1);
+		var enemy = enemys[index],
+			enemyId = enemy.id;
+		var texts = core.enemys.getSpecialHint(enemyId);
+		if (texts.length == 0) texts.push("该怪物无特殊属性。");
+		if (enemy.description) texts.push(enemy.description + "\r");
+		this._drawBookDetail_getTexts(enemy, floorId, texts);
+		texts.push("");
+		return [enemy, texts];
+	};
+},
     "一防减伤": function () {
-    // 在此增加新插件
-    ui.prototype._drawBook_drawRow3 = function (
-      index,
-      enemy,
-      top,
-      left,
-      width,
-      position
-    ) {
-      // 绘制第三行
-      core.setTextAlign("ui", "left");
-      var b13 = this._buildFont(13, true),
-        f13 = this._buildFont(13, false);
-      var col1 = left,
-        col2 = left + (width * 9) / 25,
-        col3 = left + (width * 17) / 25;
-      core.fillText("ui", "临界", col1, position, "#DDDDDD", f13);
-      core.fillText(
-        "ui",
-        core.formatBigNumber(enemy.critical || 0),
-        col1 + 30,
-        position,
-        null,
-        b13
-      );
-      core.fillText("ui", "减伤", col2, position, null, f13);
-      core.fillText(
-        "ui",
-        core.formatBigNumber(enemy.criticalDamage || 0),
-        col2 + 30,
-        position,
-        null,
-        b13
-      );
-      //core.fillText('ui', '加防', col3, position, null, f13);
-      core.fillText("ui", "1防", col3, position, null, f13);
-      core.fillText(
-        "ui",
-        core.formatBigNumber(enemy.defDamage || 0),
-        col3 + 30,
-        position,
-        null,
-        b13
-      );
-    };
-    ////// 1防减伤计算 //////
-    enemys.prototype.getDefDamage = function (enemy, k, x, y, floorId) {
-      if (typeof enemy == "string") enemy = core.material.enemys[enemy];
-      k = k || 1;
-      var nowDamage = this._getDamage(enemy, null, x, y, floorId);
-      var nextDamage = this._getDamage(
-        enemy,
-        { def: core.status.hero.def + k },
-        x,
-        y,
-        floorId
-      );
-      if (nowDamage == null || nextDamage == null) return "???";
-      return nowDamage - nextDamage;
-    };
-    //防御倍数
-    enemys.prototype._getCurrentEnemys_addEnemy_defDamage = function (
-      enemy,
-      x,
-      y,
-      floorId
-    ) {
-      var ratio = core.status.maps[floorId || core.status.floorId].ratio || 1;
-      //第一行为按照ratio值计算减防,第二行为1防减伤
-      //return this.getDefDamage(enemy, ratio, x, y, floorId);
-      return this.getDefDamage(enemy, null, x, y, floorId);
-    };
-  },
+	// 在此增加新插件
+	// 
+
+	ui.prototype._drawBook_drawRow3 = function (
+		index,
+		enemy,
+		top,
+		left,
+		width,
+		position
+	) {
+		// 绘制第三行
+		core.setTextAlign("ui", "left");
+		var b13 = this._buildFont(13, true),
+			f13 = this._buildFont(13, false);
+		var col1 = left,
+			col2 = left + (width * 9) / 25,
+			col3 = left + (width * 17) / 25;
+		core.fillText('ui', '攻击临界', col1 - 50, position, '#DDDDDD', f13);
+		core.fillText('ui', `[${core.formatBigNumber(enemy.criticalAtk?.[0] || 0)},${core.formatBigNumber(enemy.criticalAtk?.[1] || 0)}]`, col1 + 10, position, null, b13);
+		core.fillText('ui', '法强临界', col2 - 33, position, null, f13);
+		core.fillText('ui', `[${core.formatBigNumber(enemy.criticalSpell[0] || 0)},${core.formatBigNumber(enemy.criticalSpell?.[1] || 0)}]`, col2 + 27, position, null, b13);
+		//core.fillText('ui', '加防', col3, position, null, f13);
+		core.fillText("ui", "1防", col3, position, null, f13);
+		core.fillText(
+			"ui",
+			core.formatBigNumber(enemy.defDamage || 0),
+			col3 + 30,
+			position,
+			null,
+			b13
+		);
+	};
+	////// 1防减伤计算 //////
+	enemys.prototype.getDefDamage = function (enemy, k, x, y, floorId) {
+		if (typeof enemy == "string") enemy = core.material.enemys[enemy];
+		k = k || 1;
+		var nowDamage = this._getDamage(enemy, null, x, y, floorId);
+		var nextDamage = this._getDamage(
+			enemy, { def: core.status.hero.def + k },
+			x,
+			y,
+			floorId
+		);
+		if (nowDamage == null || nextDamage == null) return "???";
+		return nowDamage - nextDamage;
+	};
+	//防御倍数
+	enemys.prototype._getCurrentEnemys_addEnemy_defDamage = function (
+		enemy,
+		x,
+		y,
+		floorId
+	) {
+		var ratio = core.status.maps[floorId || core.status.floorId].ratio || 1;
+		//第一行为按照ratio值计算减防,第二行为1防减伤
+		//return this.getDefDamage(enemy, ratio, x, y, floorId);
+		return this.getDefDamage(enemy, null, x, y, floorId);
+	};
+},
     "新道具栏/装备栏": function () {
-    // 在此增加新插件
-    // 注:///// *** 裹起来的区域: 该区域内参数可以随意更改调整ui绘制 不会影响总体布局
-    // 请尽量修改该区域而不是其他区域 修改的时候最好可以对照现有ui修改
+	// 这个插件有点离谱 个人觉得参数过多只会降低可读性,还不如硬编码
 
-    ///// *** 道具类型
-    // cls对应name
-    var itemClsName = {
-      constants: "永久道具",
-      tools: "消耗道具",
-    };
-    // 一页最大放的道具数量 将把整个道具左栏分成num份 每份是一个道具项
-    var itemNum = 12;
-    ///// ***
+	// 注:///// *** 裹起来的区域: 该区域内参数可以随意更改调整ui绘制 不会影响总体布局
+	// 请尽量修改该区域而不是其他区域 修改的时候最好可以对照现有ui修改
 
-    // 背景设置
-    this.drawBoxBackground = function (ctx) {
-      core.setTextAlign(ctx, "left");
-      core.clearMap(ctx);
-      core.deleteCanvas("_selector");
-      var info = core.status.thisUIEventInfo || {};
+	///// *** 道具类型
+	// cls对应name
+	var itemClsName = {
+		constants: "永久道具",
+		tools: "消耗道具",
+	};
+	// 一页最大放的道具数量 将把整个道具左栏分成num份 每份是一个道具项
+	var itemNum = 12;
+	///// ***
 
-      ///// *** 背景设置
-      var max = core.__PIXELS__;
-      var x = 2,
-        y = x,
-        w = max - x * 2,
-        h = w;
-      var borderWidth = 2,
-        borderRadius = 5, // radius:圆角矩形的圆角半径
-        borderStyle = "#fff";
-      var backgroundColor = "gray";
-      // 设置背景不透明度(0.85)
-      var backgroundAlpha = 0.85;
-      ///// ***
+	// 背景设置
+	function drawBoxBackground(ctx) {
+		core.setTextAlign(ctx, "left");
+		core.clearMap(ctx);
+		core.deleteCanvas("_selector");
+		var info = core.status.thisUIEventInfo || {};
 
-      var start_x = x + borderWidth / 2,
-        start_y = y + borderWidth / 2,
-        width = max - start_x * 2,
-        height = max - start_y * 2;
+		///// *** 背景设置
+		var max = core.__PIXELS__;
+		var x = 2,
+			y = x,
+			w = max - x * 2,
+			h = w;
+		var borderWidth = 2,
+			borderRadius = 5, // radius:圆角矩形的圆角半径
+			borderStyle = "#fff";
+		var backgroundColor = "gray";
+		// 设置背景不透明度(0.85)
+		var backgroundAlpha = 0.85;
+		///// ***
 
-      // 渐变色背景的一个例子(黑色渐变白色):
-      // 有关渐变色的具体知识请网上搜索canvas createGradient了解
-      /*
+		var start_x = x + borderWidth / 2,
+			start_y = y + borderWidth / 2,
+			width = max - start_x * 2,
+			height = max - start_y * 2;
+
+		// 渐变色背景的一个例子(黑色渐变白色):
+		// 有关渐变色的具体知识请网上搜索canvas createGradient了解
+		/*
 		   var grd = ctx.createLinearGradient(x, y, x + w, y);
 		   grd.addColorStop(0, "black");
 		   grd.addColorStop(1, "white");
 		   backgroundColor = grd;
 		*/
-      // 使用图片背景要注释掉下面的strokeRect和fillRoundRect
-      // 图片背景的一个例子:
-      /*
+		// 使用图片背景要注释掉下面的strokeRect和fillRoundRect
+		// 图片背景的一个例子:
+		/*
 		   core.drawImage(ctx, "xxx.png", x, y, w, h);
 		   core.strokeRect(ctx, x, y, w, h, borderStyle, borderWidth);
 		*/
-      core.setAlpha(ctx, backgroundAlpha);
-      core.strokeRoundRect(
-        ctx,
-        x,
-        y,
-        w,
-        h,
-        borderRadius,
-        borderStyle,
-        borderWidth
-      );
-      core.fillRoundRect(
-        ctx,
-        start_x,
-        start_y,
-        width,
-        height,
-        borderRadius,
-        backgroundColor
-      );
-      core.setAlpha(ctx, 1);
-
-      ///// *** 左栏配置
-      var leftbar_height = height;
-      // 左边栏宽度(width*0.6) 本身仅为坐标使用 需要与底下的rightbar_width(width*0.4)同时更改
-      var leftbar_width = width * 0.6;
-      ///// ***
-
-      // xxx_right参数 代表最右侧坐标
-      var leftbar_right = start_x + leftbar_width - borderWidth / 2;
-      var leftbar_bottom = start_y + leftbar_height;
-      var leftbar_x = start_x;
-      var leftbar_y = start_y;
-
-      ///// *** 道具栏配置
-      var boxName_color = "#fff";
-      var boxName_fontSize = 15;
-      var boxName_font = core.ui._buildFont(boxName_fontSize, true);
-      var arrow_x = 10 + start_x;
-      var arrow_y = 10 + start_y;
-      var arrow_width = 20;
-      var arrow_style = "white";
-      // 暂时只能是1 否则不太行 等待新样板(2.7.3)之后对drawArrow做优化
-      var arrow_lineWidth = 1;
-      // 右箭头
-      var rightArrow_right = leftbar_right - 10;
-      // 道具内栏顶部坐标 本质是通过该项 控制(道具栏顶部文字和箭头)与道具内栏顶部的间隔
-      var itembar_top = arrow_y + 15;
-      ///// ***
-
-      var itembar_right = rightArrow_right;
-      var boxName =
-        core.status.event.id == "toolbox"
-          ? "\r[yellow]道具栏\r | 装备栏"
-          : "道具栏 | \r[yellow]装备栏\r";
-      core.drawArrow(
-        ctx,
-        arrow_x + arrow_width,
-        arrow_y,
-        arrow_x,
-        arrow_y,
-        arrow_style,
-        arrow_lineWidth
-      );
-      core.drawArrow(
-        ctx,
-        rightArrow_right - arrow_width,
-        arrow_y,
-        rightArrow_right,
-        arrow_y,
-        arrow_style,
-        arrow_lineWidth
-      );
-      core.setTextAlign(ctx, "center");
-      core.setTextBaseline(ctx, "middle");
-      var changeBox = function () {
-        var id = core.status.event.id;
-        core.closePanel();
-        if (id == "toolbox") core.openEquipbox();
-        else core.openToolbox();
-      };
-      core.fillText(
-        ctx,
-        boxName,
-        (leftbar_right + leftbar_x) / 2,
-        arrow_y + 2,
-        boxName_color,
-        boxName_font
-      );
-
-      ///// *** 底栏按钮
-      var pageBtn_radius = 8;
-      // xxx_left 最左侧坐标
-      var pageBtn_left = leftbar_x + 3;
-      var pageBtn_right = leftbar_right - 3;
-      // xxx_bottom 最底部坐标
-      var pageBtn_bottom = leftbar_bottom - 2;
-      var pageBtn_borderStyle = "#fff";
-      var pageBtn_borderWidth = 2;
-      var pageText_color = "#fff";
-      // 底部按钮与上面的道具内栏的间隔大小
-      var bottomSpace = 8;
-      ///// ***
-
-      core.drawItemListbox_setPageBtn(
-        ctx,
-        pageBtn_left,
-        pageBtn_right,
-        pageBtn_bottom,
-        pageBtn_radius,
-        pageBtn_borderStyle,
-        pageBtn_borderWidth
-      );
-      var page = info.page || 1;
-      var pageFontSize = pageBtn_radius * 2 - 4;
-      var pageFont = core.ui._buildFont(pageFontSize);
-      core.setPageItems(page);
-      var num = itemNum;
-      if (core.status.event.id == "equipbox") num -= 5;
-      var maxPage = info.maxPage;
-      var pageText = page + " / " + maxPage;
-      core.setTextAlign(ctx, "center");
-      core.setTextBaseline(ctx, "bottom");
-      core.fillText(
-        ctx,
-        pageText,
-        (leftbar_x + leftbar_right) / 2,
-        pageBtn_bottom,
-        pageText_color,
-        pageFont
-      );
-      core.addUIEventListener(
-        start_x,
-        start_y,
-        leftbar_right - start_x,
-        arrow_y - start_y + 13,
-        changeBox
-      );
-      var itembar_height = Math.ceil(
-        pageBtn_bottom -
-          pageBtn_radius * 2 -
-          pageBtn_borderWidth / 2 -
-          bottomSpace -
-          itembar_top
-      );
-      var oneItemHeight = (itembar_height - 4) / itemNum;
-      return {
-        x: start_x,
-        y: start_y,
-        width: width,
-        height: height,
-        leftbar_right: leftbar_right,
-        obj: {
-          x: arrow_x,
-          y: itembar_top,
-          width: itembar_right - arrow_x,
-          height: itembar_height,
-          oneItemHeight: oneItemHeight,
-        },
-      };
-    };
-
-    this.drawItemListbox = function (ctx, obj) {
-      ctx = ctx || core.canvas.ui;
-      var itembar_x = obj.x,
-        itembar_y = obj.y,
-        itembar_width = obj.width,
-        itembar_height = obj.height,
-        itemNum = obj.itemNum,
-        oneItemHeight = obj.oneItemHeight;
-      var itembar_right = itembar_x + itembar_width;
-      var info = core.status.thisUIEventInfo || {};
-      var obj = {};
-      var page = info.page || 1,
-        index = info.index,
-        select = info.select || {};
-
-      ///// *** 道具栏内栏配置
-      var itembar_style = "black";
-      var itembar_alpha = 0.7;
-      // 一个竖屏下减少道具显示的例子:
-      // if (core.domStyle.isVertical) itemNum = 10;
-      // 每个道具项的上下空隙占总高度的比例
-      var itembar_marginHeightRatio = 0.2;
-      // 左右间隔空隙
-      var item_marginLeft = 2;
-      var item_x = itembar_x + 2,
-        item_y = itembar_y + 2,
-        item_right = itembar_right - 2,
-        itemName_color = "#fff";
-      // 修改此项以更换闪烁光标
-      var item_selector = "winskin.webp";
-      ///// ***
-
-      core.setAlpha(ctx, itembar_alpha);
-      core.fillRect(
-        ctx,
-        itembar_x,
-        itembar_y,
-        itembar_width,
-        itembar_height,
-        itembar_style
-      );
-      core.setAlpha(ctx, 1);
-      var pageItems = core.setPageItems(page);
-      var marginHeight = itembar_marginHeightRatio * oneItemHeight;
-      core.setTextBaseline(ctx, "middle");
-      var originColor = itemName_color;
-      for (var i = 0; i < pageItems.length; i++) {
-        itemName_color = originColor;
-        var item = pageItems[i];
-        // 设置某个的字体颜色的一个例子
-        // if (item.id == "xxx") itemName_color = "green";
-        core.drawItemListbox_drawItem(
-          ctx,
-          item_x,
-          item_right,
-          item_y,
-          oneItemHeight,
-          item_marginLeft,
-          marginHeight,
-          itemName_color,
-          pageItems[i]
-        );
-        if (index == i + 1)
-          core.ui._drawWindowSelector(
-            item_selector,
-            item_x + 1,
-            item_y - 1,
-            item_right - item_x - 2,
-            oneItemHeight - 2
-          );
-        item_y += oneItemHeight;
-      }
-    };
-
-    this.drawToolboxRightbar = function (ctx, obj) {
-      ctx = ctx || core.canvas.ui;
-      var info = core.status.thisUIEventInfo || {};
-      var page = info.page || 1,
-        index = info.index || 1,
-        select = info.select || {};
-      var start_x = obj.x,
-        start_y = obj.y,
-        width = obj.width,
-        height = obj.height;
-      var toolboxRight = start_x + width,
-        toolboxBottom = start_y + height;
-
-      ///// *** 侧边栏(rightbar)背景设置(物品介绍)
-      var rightbar_width = width * 0.4;
-      var rightbar_height = height;
-      var rightbar_lineWidth = 2;
-      var rightbar_lineStyle = "#fff";
-      ///// ***
-
-      var rightbar_x = toolboxRight - rightbar_width - rightbar_lineWidth / 2;
-      var rightbar_y = start_y;
-      core.drawLine(
-        ctx,
-        rightbar_x,
-        rightbar_y,
-        rightbar_x,
-        rightbar_y + rightbar_height,
-        rightbar_lineStyle,
-        rightbar_lineWidth
-      );
-
-      // 获取道具id(有可能为null)
-      var itemId = select.id;
-      var item = core.material.items[itemId];
-
-      ///// *** 侧边栏物品Icon信息
-      var iconRect_y = rightbar_y + 10;
-      // space:间距
-      // 这里布局设定iconRect与侧边栏左边框 itemName与工具栏右边框 itemRect与itemName的间距均为space
-      var space = 15;
-      var iconRect_x = rightbar_x + space;
-      var iconRect_radius = 2,
-        iconRect_width = 32,
-        iconRect_height = 32,
-        iconRect_style = "#fff",
-        iconRect_lineWidth = 2;
-      ///// ***
-
-      var iconRect_bottom = iconRect_y + iconRect_height,
-        iconRect_right = iconRect_x + iconRect_width;
-
-      ///// *** 侧边栏各项信息
-      var itemTextFontSize = 15,
-        itemText_x = iconRect_x - 4,
-        itemText_y = Math.floor(start_y + rightbar_height * 0.25), // 坐标取整防止模糊
-        itemClsFontSize = 15,
-        itemClsFont = core.ui._buildFont(itemClsFontSize),
-        itemClsColor = "#fff",
-        itemCls_x = itemText_x - itemClsFontSize / 2,
-        itemCls_middle = (iconRect_bottom + itemText_y) / 2, //_middle代表文字的中心y坐标
-        itemNameFontSize = 18,
-        itemNameColor = "#fff",
-        itemNameFont = core.ui._buildFont(itemNameFontSize, true);
-      var itemName_x = iconRect_right + space;
-      var itemName_middle =
-        iconRect_y + iconRect_height / 2 + iconRect_lineWidth;
-      // 修改这里可以编辑未选中道具时的默认值
-      var defaultItem = {
-        cls: "constants",
-        name: "未知道具",
-        text: "没有道具最永久",
-      };
-      var defaultEquip = {
-        cls: "equips",
-        name: "未知装备",
-        text: "一无所有,又何尝不是一种装备",
-        equipCls: "无",
-        equip: {
-          type: "装备",
-        },
-      };
-      ///// ***
-
-      var originItem = item;
-      if (core.status.event.id == "equipbox") item = item || defaultEquip;
-      item = item || defaultItem;
-      var itemCls = item.cls,
-        itemName = item.name,
-        itemText = item.text;
-      itemText = core.replaceText(itemText);
-      if (itemText[0] == "," || itemText[0] == ",")
-        itemText = itemText.substring(1);
-
-      /* 一个根据道具id修改道具名字(右栏)的例子
-       * if (item.id == "xxx") itemNameColor = "red";
-       */
-      var itemClsName = core.getItemClsName(item);
-      var itemNameMaxWidth =
-        rightbar_width - iconRect_width - iconRect_lineWidth * 2 - space * 2;
-      core.strokeRoundRect(
-        ctx,
-        iconRect_x,
-        iconRect_y,
-        iconRect_width,
-        iconRect_height,
-        iconRect_radius,
-        iconRect_style,
-        iconRect_lineWidth
-      );
-      if (item.id)
-        core.drawIcon(
-          ctx,
-          item.id,
-          iconRect_x + iconRect_lineWidth / 2,
-          iconRect_y + iconRect_lineWidth / 2,
-          iconRect_width - iconRect_lineWidth,
-          iconRect_height - iconRect_lineWidth
-        );
-      core.setTextAlign(ctx, "left");
-      core.setTextBaseline(ctx, "middle");
-      if (itemCls === "equips" && item.id) {
-        itemName = "【" + item.equipCls + "】" + itemName;
-      }
-      core.fillText(
-        ctx,
-        itemName,
-        itemName_x,
-        itemName_middle,
-        itemNameColor,
-        itemNameFont,
-        itemNameMaxWidth
-      );
-      if (!item.equip)
-        core.fillText(
-          ctx,
-          "【" + itemClsName + "】",
-          itemCls_x,
-          itemCls_middle,
-          itemClsColor,
-          itemClsFont
-        );
-
-      var statusText = "";
-      if (core.status.event.id == "equipbox") {
-        var type = item.equip.type;
-        if (typeof type == "string") type = core.getEquipTypeByName(type);
-        var compare = core.compareEquipment(item.id, core.getEquip(type));
-        var compare2;
-        if (item.equipCls === "双手剑")
-          compare2 = core.compareEquipment(null, core.getEquip(1));
-        if (info.select.action == "unload")
-          compare = core.compareEquipment(null, item.id);
-        // --- 变化值...
-        for (var name in core.status.hero) {
-          if (typeof core.status.hero[name] != "number") continue;
-          var nowValue = core.getRealStatus(name);
-          // 查询新值
-          var newValue = Math.floor(
-            ((core.getStatus(name) +
-              (compare.value[name] || 0) +
-              (compare2?.value[name] || 0)) *
-              (core.getBuff(name) * 100 +
-                (compare.percentage[name] || 0) +
-                (compare2?.percentage[name] || 0))) /
-              100
-          );
-          if (nowValue == newValue) continue;
-          var color = newValue > nowValue ? "#00FF00" : "#FF0000";
-          nowValue = core.formatBigNumber(nowValue);
-          newValue = core.formatBigNumber(newValue);
-          statusText +=
-            core.getStatusLabel(name) +
-            " " +
-            nowValue +
-            "->\r[" +
-            color +
-            "]" +
-            newValue +
-            "\r\n";
-        }
-      }
-      itemText = statusText + itemText;
-      if (item.equip) {
-        core.drawTextContent(ctx, itemText, {
-          left: itemText_x,
-          top: itemCls_middle,
-          bold: false,
-          color: "white",
-          align: "left",
-          fontSize: itemTextFontSize,
-          maxWidth:
-            rightbar_width -
-            (itemText_x - rightbar_x) * 2 +
-            itemTextFontSize / 2,
-        });
-      } else {
-        core.drawTextContent(ctx, itemText, {
-          left: itemText_x,
-          top: itemText_y,
-          bold: false,
-          color: "white",
-          align: "left",
-          fontSize: itemTextFontSize,
-          maxWidth:
-            rightbar_width -
-            (itemText_x - rightbar_x) * 2 +
-            itemTextFontSize / 2,
-        });
-      }
-
-      ///// *** 退出按钮设置
-      var btnRadius = 10;
-      var btnBorderWidth = 2;
-      var btnRight = toolboxRight - 2;
-      var btnBottom = toolboxBottom - 2;
-      var btnBorderStyle = "#fff";
-      ///// ***
-
-      // 获取圆心位置
-      var btn_x = btnRight - btnRadius - btnBorderWidth / 2,
-        btn_y = btnBottom - btnRadius - btnBorderWidth / 2;
-      core.drawToolbox_setExitBtn(
-        ctx,
-        btn_x,
-        btn_y,
-        btnRadius,
-        btnBorderStyle,
-        btnBorderWidth
-      );
-
-      ///// *** 使用按钮设置
-      var useBtnHeight = btnRadius * 2;
-      // 这里不设置useBtnWidth而是根据各项数据自动得出width
-      var useBtnRadius = useBtnHeight / 2;
-      var useBtn_x = rightbar_x + 4,
-        useBtn_y = btnBottom - useBtnHeight;
-      var useBtnBorderStyle = "#fff";
-      var useBtnBorderWidth = btnBorderWidth;
-      ///// ***
-
-      core.drawToolbox_setUseBtn(
-        ctx,
-        useBtn_x,
-        useBtn_y,
-        useBtnRadius,
-        useBtnHeight,
-        useBtnBorderStyle,
-        useBtnBorderWidth
-      );
-    };
-
-    this.drawEquipbox_drawOthers = function (ctx, obj) {
-      var info = core.status.thisUIEventInfo;
-
-      ///// *** 装备格设置
-      var equipList_lineWidth = 2;
-      var equipList_boxSize = 32;
-      var equipList_borderWidth = 2;
-      var equipList_borderStyle = "#fff";
-      var equipList_nameColor = "#fff";
-      ///// ***
-
-      var equipList_x = obj.x + 4,
-        equipList_bottom = obj.obj.y - equipList_lineWidth,
-        equipList_y = equipList_bottom - obj.obj.oneItemHeight * reduceItem - 2,
-        equipList_height = equipList_bottom - equipList_y;
-      var equipList_right = obj.leftbar_right,
-        equipList_width = equipList_right - equipList_x;
-      core.drawLine(
-        ctx,
-        obj.x,
-        equipList_bottom + equipList_lineWidth / 2,
-        equipList_right,
-        equipList_bottom + equipList_lineWidth / 2,
-        equipList_borderStyle,
-        equipList_lineWidth
-      );
-      var toDrawList = core.status.globalAttribute.equipName,
-        len = toDrawList.length;
-
-      ///// *** 装备格设置
-      var maxItem = 2;
-      var box_width = 32,
-        box_height = 32,
-        box_borderStyle = "#fff",
-        box_selectBorderStyle = "gold", // 选中的装备格的颜色
-        box_borderWidth = 2;
-      var boxName_fontSize = 14,
-        boxName_space = 2,
-        boxName_color = "#fff"; // 装备格名称与上面的装备格框的距离
-      var maxLine = Math.ceil(len / maxItem);
-      ///// ***
-      var l = Math.sqrt(len);
-      if (Math.pow(l) == len && len != 4) {
-        if (l <= maxItem) maxItem = l;
-      }
-      maxItem = Math.min(toDrawList.length, maxItem);
-      info.equips = maxItem;
-
-      var boxName_font = core.ui._buildFont(boxName_fontSize);
-      // 总宽高减去所有装备格宽高得到空隙大小
-      var oneBoxWidth = box_width + box_borderWidth * 2;
-      var oneBoxHeight =
-        box_height + boxName_fontSize + boxName_space + 2 * box_borderWidth;
-      var space_y = (equipList_height - maxLine * oneBoxHeight) / (1 + maxLine),
-        space_x = (equipList_width - maxItem * oneBoxWidth) / (1 + maxItem);
-      var box_x = equipList_x + space_x,
-        box_y = equipList_y + space_y + 12;
-      for (var i = 0; i < 2; i++) {
-        var id = core.getEquip(i),
-          name = toDrawList[i];
-        if (i === 0) name = "主手";
-        if (i === 1) name = "副手";
-        var selectBorder = false;
-        if (core.status.thisUIEventInfo.select.type == i) selectBorder = true;
-        var borderStyle = selectBorder
-          ? box_selectBorderStyle
-          : box_borderStyle;
-        core.drawEquipbox_drawOne(
-          ctx,
-          name,
-          id,
-          box_x,
-          box_y,
-          box_width,
-          box_height,
-          boxName_space,
-          boxName_font,
-          boxName_color,
-          borderStyle,
-          box_borderWidth
-        );
-        var todo = new Function(
-          "core.clickOneEquipbox('" + id + "'," + i + ")"
-        );
-        core.addUIEventListener(
-          box_x - box_borderWidth / 2,
-          box_y - box_borderWidth / 2,
-          oneBoxWidth,
-          oneBoxHeight,
-          todo
-        );
-        box_x += space_x + oneBoxWidth;
-        if ((i + 1) % maxItem == 0) {
-          box_x = equipList_x + space_x;
-          box_y += space_y + oneBoxHeight;
-        }
-      }
-      if (core.material.items[core.getEquip(0)]?.equipCls === "双手剑") {
-        core.drawLine(
-          ctx,
-          equipList_x + space_x + space_x + oneBoxWidth,
-          equipList_y + space_y + 12,
-          equipList_x +
-            space_x +
-            space_x +
-            oneBoxWidth +
-            box_width +
-            box_borderWidth,
-          equipList_y + space_y + box_height + 12
-        );
-        core.drawLine(
-          ctx,
-          equipList_x + space_x + space_x + oneBoxWidth,
-          equipList_y + space_y + box_height + 12,
-          equipList_x +
-            space_x +
-            space_x +
-            oneBoxWidth +
-            box_width +
-            box_borderWidth,
-          equipList_y + space_y + 12
-        );
-      }
-      ///// *** 装备格设置
-      var maxItem = 3;
-      var box_width = 32,
-        box_height = 32,
-        box_borderStyle = "#fff",
-        box_selectBorderStyle = "gold", // 选中的装备格的颜色
-        box_borderWidth = 2;
-      var boxName_fontSize = 14,
-        boxName_space = 2,
-        boxName_color = "#fff"; // 装备格名称与上面的装备格框的距离
-      var maxLine = Math.ceil(len / maxItem);
-      ///// ***
-      var l = Math.sqrt(len);
-      if (Math.pow(l) == len && len != 4) {
-        if (l <= maxItem) maxItem = l;
-      }
-      maxItem = Math.min(toDrawList.length, maxItem);
-      info.equips = maxItem;
-
-      var boxName_font = core.ui._buildFont(boxName_fontSize);
-      // 总宽高减去所有装备格宽高得到空隙大小
-      var oneBoxWidth = box_width + box_borderWidth * 2;
-      var oneBoxHeight =
-        box_height + boxName_fontSize + boxName_space + 2 * box_borderWidth;
-      var space_y = (equipList_height - maxLine * oneBoxHeight) / (1 + maxLine),
-        space_x = (equipList_width - maxItem * oneBoxWidth) / (1 + maxItem);
-      var box_x = equipList_x + space_x,
-        box_y = equipList_y + space_y + space_y + oneBoxHeight;
-      for (var i = 2; i < len; i++) {
-        var id = core.getEquip(i),
-          name = toDrawList[i];
-        var selectBorder = false;
-        if (core.status.thisUIEventInfo.select.type == i) selectBorder = true;
-        var borderStyle = selectBorder
-          ? box_selectBorderStyle
-          : box_borderStyle;
-        core.drawEquipbox_drawOne(
-          ctx,
-          name,
-          id,
-          box_x,
-          box_y,
-          box_width,
-          box_height,
-          boxName_space,
-          boxName_font,
-          boxName_color,
-          borderStyle,
-          box_borderWidth
-        );
-        var todo = new Function(
-          "core.clickOneEquipbox('" + id + "'," + i + ")"
-        );
-        core.addUIEventListener(
-          box_x - box_borderWidth / 2,
-          box_y - box_borderWidth / 2,
-          oneBoxWidth,
-          oneBoxHeight,
-          todo
-        );
-        box_x += space_x + oneBoxWidth;
-      }
-    };
-
-    this.drawToolbox = function (ctx) {
-      ctx = ctx || core.canvas.ui;
-      core.status.thisEventClickArea = [];
-
-      var info = core.drawBoxBackground(ctx);
-      info.itemNum = itemNum;
-      core.drawItemListbox(ctx, info.obj);
-      core.drawToolboxRightbar(ctx, info);
-      core.setTextBaseline(ctx, "alphabetic");
-      core.setTextAlign("left");
-    };
-
-    var reduceItem = 4;
-    this.drawEquipbox = function (ctx) {
-      ctx = ctx || core.canvas.ui;
-      core.status.thisEventClickArea = [];
-      var info = core.drawBoxBackground(ctx);
-      info.itemNum = itemNum - reduceItem;
-      info.obj.y += info.obj.oneItemHeight * reduceItem;
-      info.obj.height -= info.obj.oneItemHeight * reduceItem;
-      core.drawItemListbox(ctx, info.obj);
-      core.drawEquipbox_drawOthers(ctx, info);
-      core.drawToolboxRightbar(ctx, info);
-      core.setTextBaseline(ctx, "alphabetic");
-      core.setTextAlign("left");
-    };
-
-    this.drawEquipbox_drawOne = function (
-      ctx,
-      name,
-      id,
-      x,
-      y,
-      width,
-      height,
-      space,
-      font,
-      color,
-      style,
-      lineWidth
-    ) {
-      if (id)
-        core.drawIcon(
-          ctx,
-          id,
-          x + lineWidth / 2,
-          y + lineWidth / 2,
-          width,
-          height
-        );
-      core.strokeRect(
-        ctx,
-        x,
-        y,
-        width + lineWidth,
-        height + lineWidth,
-        style,
-        lineWidth
-      );
-      core.setTextAlign(ctx, "center");
-      core.setTextBaseline(ctx, "top");
-      var tx = (x + x + lineWidth / 2 + width) / 2,
-        ty = y + height + (lineWidth / 2) * 3 + space;
-      core.fillText(ctx, name, tx, ty, color, font);
-      core.setTextBaseline(ctx, "alphabetic");
-      core.setTextAlign("left");
-    };
-
-    this.drawItemListbox_drawItem = function (
-      ctx,
-      left,
-      right,
-      top,
-      height,
-      marginLeft,
-      marginHeight,
-      style,
-      id
-    ) {
-      var info = core.status.thisUIEventInfo;
-      var nowClick = info.index;
-      var item = core.material.items[id] || {};
-      var name = item.name || "???";
-      var num = core.itemCount(id) || 0;
-      var fontSize = Math.floor(height - marginHeight * 2);
-      core.setTextAlign(ctx, "right");
-      var numText = "x" + num;
-      core.fillText(
-        ctx,
-        numText,
-        right - marginLeft,
-        top + height / 2,
-        style,
-        core.ui._buildFont(fontSize)
-      );
-      if (name != "???")
-        core.drawIcon(
-          ctx,
-          id,
-          left + marginLeft,
-          top + marginHeight,
-          fontSize,
-          fontSize
-        );
-      var text_x = left + marginLeft + fontSize + 2;
-      var maxWidth = right - core.calWidth(ctx, numText) - text_x;
-      core.setTextAlign(ctx, "left");
-      core.fillText(
-        ctx,
-        name,
-        text_x,
-        top + height / 2,
-        style,
-        core.ui._buildFont(fontSize),
-        maxWidth
-      );
-      var todo = new Function("core.clickItemFunc('" + id + "');");
-      core.addUIEventListener(left, top, right - left, height, todo);
-    };
-
-    this.setPageItems = function (page) {
-      var num = itemNum;
-      if (core.status.event.id == "equipbox") num -= reduceItem;
-      var info = core.status.thisUIEventInfo;
-      if (!info) return;
-      page = page || info.page;
-      var items = core.getToolboxItems(
-        core.status.event.id == "toolbox" ? "all" : "equips"
-      );
-      info.allItems = items;
-      var maxPage = Math.ceil(items.length / num);
-      info.maxPage = maxPage;
-      var pageItems = items.slice((page - 1) * num, page * num);
-      info.pageItems = pageItems;
-      info.maxItem = pageItems.length;
-      if (items.length == 0 && pageItems.length == 0) info.index = null;
-      if (pageItems.length == 0 && info.page > 1) {
-        info.page = Math.max(1, info.page - 1);
-        return core.setPageItems(info.page);
-      }
-      return pageItems;
-    };
-
-    this.drawToolbox_setExitBtn = function (ctx, x, y, r, style, lineWidth) {
-      core.strokeCircle(ctx, x, y, r, style, lineWidth);
-      ctx.textAlign = "center";
-      ctx.textBaseline = "middle";
-      var textSize = Math.sqrt(2) * r;
-      core.fillText(
-        ctx,
-        "x",
-        x,
-        y,
-        style,
-        core.ui._buildFont(textSize),
-        textSize
-      );
-      core.setTextAlign(ctx, "start");
-      core.setTextBaseline(ctx, "top");
-
-      var todo = function () {
-        core.closePanel();
-      };
-      core.addUIEventListener(x - r, y - r, r * 2, r * 2, todo);
-    };
-
-    this.drawToolbox_setUseBtn = function (ctx, x, y, r, h, style, lineWidth) {
-      core.setTextAlign(ctx, "left");
-      core.setTextBaseline(ctx, "top");
-      var fontSize = h - 4;
-      var font = core.ui._buildFont(fontSize);
-      var text = core.status.event.id == "toolbox" ? "使用" : "装备";
-      if (core.status.thisUIEventInfo.select.action == "unload") text = "卸下";
-      var w = core.calWidth(ctx, text, font) + 2 * r + lineWidth / 2;
-
-      core.strokeRoundRect(ctx, x, y, w, h, r, style, lineWidth);
-      core.fillText(ctx, text, x + r, y + lineWidth / 2 + 2, style, font);
-
-      var todo = function () {
-        core.useSelectItemInBox();
-      };
-      core.addUIEventListener(x, y, w, h, todo);
-    };
-
-    this.drawItemListbox_setPageBtn = function (
-      ctx,
-      left,
-      right,
-      bottom,
-      r,
-      style,
-      lineWidth
-    ) {
-      var offset = lineWidth / 2 + r;
-
-      var x = left + offset;
-      var y = bottom - offset;
-      var pos = (Math.sqrt(2) / 2) * (r - lineWidth / 2);
-      core.fillPolygon(
-        ctx,
-        [
-          [x - pos, y],
-          [x + pos - 2, y - pos],
-          [x + pos - 2, y + pos],
-        ],
-        style
-      );
-      core.strokeCircle(ctx, x, y, r, style, lineWidth);
-      var todo = function () {
-        core.addItemListboxPage(-1);
-      };
-      core.addUIEventListener(x - r - 2, y - r - 2, r * 2 + 4, r * 2 + 4, todo);
-
-      x = right - offset;
-      core.fillPolygon(
-        ctx,
-        [
-          [x + pos, y],
-          [x - pos + 2, y - pos],
-          [x - pos + 2, y + pos],
-        ],
-        style
-      );
-      core.strokeCircle(ctx, x, y, r, style, lineWidth);
-      var todo = function () {
-        core.addItemListboxPage(1);
-      };
-      core.addUIEventListener(x - r - 2, y - r - 2, r * 2 + 4, r * 2 + 4, todo);
-    };
-
-    this.clickItemFunc = function (id) {
-      var info = core.status.thisUIEventInfo;
-      if (!info) return;
-      if (info.select.id == id) return core.useSelectItemInBox();
-      info.select = {};
-      info.select.id = id;
-      core.setIndexAndSelect("index");
-      core.refreshBox();
-    };
-
-    this.clickOneEquipbox = function (id, type) {
-      var info = core.status.thisUIEventInfo;
-      if (!info) return;
-      if (info.select.id == id && info.select.type == type)
-        core.useSelectItemInBox();
-      else
-        core.status.thisUIEventInfo.select = {
-          id: id,
-          type: type,
-          action: "unload",
-        };
-      return core.refreshBox();
-    };
-
-    core.ui.getToolboxItems = function (cls) {
-      var list = Object.keys(core.status.hero.items[cls] || {});
-      if (cls == "all") {
-        for (var name in core.status.hero.items) {
-          if (name == "equips") continue;
-          list = list.concat(Object.keys(core.status.hero.items[name]));
-        }
-        return list
-          .filter(function (id) {
-            return !core.material.items[id].hideInToolbox;
-          })
-          .sort();
-      }
-
-      if (this.uidata.getToolboxItems) {
-        return this.uidata.getToolboxItems(cls);
-      }
-      return list
-        .filter(function (id) {
-          return !core.material.items[id].hideInToolbox;
-        })
-        .sort();
-    };
-
-    this.useSelectItemInBox = function () {
-      var info = core.status.thisUIEventInfo;
-      if (!info) return;
-      if (!info.select.id) return;
-      var id = info.select.id;
-      if (core.status.event.id == "toolbox") {
-        core.events.tryUseItem(id);
-        // core.closePanel();
-      } else if (core.status.event.id == "equipbox") {
-        var action = info.select.action || "load";
-        info.index = 1;
-        if (action == "load") {
-          var type = core.getEquipTypeById(id);
-          let equipClsid = core.material.items[id]?.equipCls;
-          let equipCls0 = core.material.items[core.getEquip(0)]?.equipCls;
-          let equipCls1 = core.material.items[core.getEquip(1)]?.equipCls;
-          if (equipClsid === "双手剑") {
-            core.unloadEquip(0, function () {
-              core.status.route.push("unEquip:" + 0);
-            });
-            core.unloadEquip(1, function () {
-              core.status.route.push("unEquip:" + 1);
-            });
-          }
-          core.loadEquip(id, function () {
-            core.status.route.push("equip:" + id);
-            info.select.type = type;
-            core.setIndexAndSelect("select");
-            core.drawEquipbox();
-          });
-        } else {
-          var type = info.select.type;
-          core.unloadEquip(type, function () {
-            core.status.route.push("unEquip:" + type);
-            info.select.type = type;
-            //info.select.action = 'load'
-            core.setIndexAndSelect("select");
-            core.drawEquipbox();
-          });
-        }
-      }
-      core.updateStatusBar();
-    };
-    this.setIndexAndSelect = function (toChange) {
-      var info = core.status.thisUIEventInfo;
-      if (!info) return;
-      core.setPageItems(info.page);
-      var index = info.index || 1;
-      var items = info.pageItems;
-      if (info.select.type != null) {
-        var type = info.select.type;
-        id = core.getEquip(type);
-        info.index = null;
-        info.select = {
-          id: id,
-          action: "unload",
-          type: type,
-        };
-        return;
-      } else {
-        info.select.action = null;
-        info.select.type = null;
-        if (toChange == "index") info.index = items.indexOf(info.select.id) + 1;
-        info.select.id = items[info.index - 1];
-      }
-    };
-
-    this.addItemListboxPage = function (num) {
-      var info = core.status.thisUIEventInfo;
-      if (!info) return;
-      var maxPage = info.maxPage || 1;
-      info.page = info.page || 1;
-      info.page += num;
-      if (info.page <= 0) info.page = maxPage;
-      if (info.page > maxPage) info.page = 1;
-      info.index = 1;
-      core.setPageItems(info.page);
-      core.setIndexAndSelect("select");
-      core.refreshBox();
-    };
-
-    this.addItemListboxIndex = function (num) {
-      var info = core.status.thisUIEventInfo;
-      if (!info) return;
-      var maxItem = info.maxItem || 0;
-      info.index = info.index || 0;
-      info.index += num;
-      if (info.index <= 0) info.index = 1;
-      if (info.index > maxItem) info.index = maxItem;
-      core.setIndexAndSelect("select");
-      core.refreshBox();
-    };
-
-    this.addEquipboxType = function (num) {
-      var info = core.status.thisUIEventInfo;
-      var type = info.select.type;
-      if (type == null && num > 0) info.select.type = 0;
-      else info.select.type = type + num;
-      var max = core.status.globalAttribute.equipName.length;
-      if (info.select.type >= max) {
-        info.select = {};
-        core.setIndexAndSelect("select");
-        return core.addItemListboxPage(0);
-      } else {
-        var m = Math.abs(info.select.type);
-        if (info.select.type < 0) info.select.type = max - m;
-        core.setIndexAndSelect("select");
-        core.refreshBox();
-        return;
-      }
-    };
-
-    core.actions._keyDownToolbox = function (keycode) {
-      if (!core.status.thisEventClickArea) return;
-      if (keycode == 37) {
-        // left
-        core.addItemListboxPage(-1);
-        return;
-      }
-      if (keycode == 38) {
-        // up
-        core.addItemListboxIndex(-1);
-        return;
-      }
-      if (keycode == 39) {
-        // right
-        core.addItemListboxPage(1);
-        return;
-      }
-      if (keycode == 40) {
-        // down
-        core.addItemListboxIndex(1);
-        return;
-      }
-    };
-
-    ////// 工具栏界面时,放开某个键的操作 //////
-    core.actions._keyUpToolbox = function (keycode) {
-      if (keycode == 81) {
-        core.ui.closePanel();
-        if (core.isReplaying()) core.control._replay_equipbox();
-        else core.openEquipbox();
-        return;
-      }
-      if (keycode == 84 || keycode == 27 || keycode == 88) {
-        core.closePanel();
-        return;
-      }
-      if (keycode == 13 || keycode == 32 || keycode == 67) {
-        var info = core.status.thisUIEventInfo;
-        if (info.select) {
-          core.useSelectItemInBox();
-        }
-        return;
-      }
-    };
-
-    core.actions._keyDownEquipbox = function (keycode) {
-      if (!core.status.thisEventClickArea) return;
-      if (keycode == 37) {
-        // left
-        var info = core.status.thisUIEventInfo;
-        if (info.index != null) return core.addItemListboxPage(-1);
-        return core.addEquipboxType(-1);
-      }
-      if (keycode == 38) {
-        // up
-        var info = core.status.thisUIEventInfo;
-        if (info.index == 1) {
-          info.select.type = core.status.globalAttribute.equipName.length - 1;
-          core.setIndexAndSelect();
-          return core.refreshBox();
-        }
-        if (info.index) return core.addItemListboxIndex(-1);
-        return core.addEquipboxType(-1 * info.equips);
-      }
-      if (keycode == 39) {
-        // right
-        var info = core.status.thisUIEventInfo;
-        if (info.index != null) return core.addItemListboxPage(1);
-        return core.addEquipboxType(1);
-      }
-      if (keycode == 40) {
-        // down
-        var info = core.status.thisUIEventInfo;
-        if (info.index) return core.addItemListboxIndex(1);
-        return core.addEquipboxType(info.equips);
-      }
-    };
-
-    core.actions._keyUpEquipbox = function (keycode, altKey) {
-      if (altKey && keycode >= 48 && keycode <= 57) {
-        core.items.quickSaveEquip(keycode - 48);
-        return;
-      }
-      if (keycode == 84) {
-        core.ui.closePanel();
-        if (core.isReplaying()) core.control._replay_toolbox();
-        else core.openToolbox();
-        return;
-      }
-      if (keycode == 81 || keycode == 27 || keycode == 88) {
-        core.closePanel();
-        return;
-      }
-      if (keycode == 13 || keycode == 32 || keycode == 67) {
-        var info = core.status.thisUIEventInfo;
-        if (info.select) core.useSelectItemInBox();
-        return;
-      }
-    };
-
-    core.registerAction(
-      "ondown",
-      "inEventClickAction",
-      function (x, y, px, py) {
-        if (!core.status.thisEventClickArea) return false;
-        // console.log(px + "," + py);
-        var info = core.status.thisEventClickArea;
-        for (var i = 0; i < info.length; i++) {
-          var obj = info[i];
-          if (
-            px >= obj.x &&
-            px <= obj.x + obj.width &&
-            py > obj.y &&
-            py < obj.y + obj.height
-          ) {
-            if (obj.todo) obj.todo();
-            break;
-          }
-        }
-        return true;
-      },
-      51
-    );
-    core.registerAction(
-      "onclick",
-      "stopClick",
-      function () {
-        if (core.status.thisEventClickArea) return true;
-      },
-      51
-    );
-
-    this.addUIEventListener = function (x, y, width, height, todo) {
-      if (!core.status.thisEventClickArea) return;
-      var obj = {
-        x: x,
-        y: y,
-        width: width,
-        height: height,
-        todo: todo,
-      };
-      core.status.thisEventClickArea.push(obj);
-    };
-
-    this.initThisEventInfo = function () {
-      core.status.thisUIEventInfo = {
-        page: 1,
-        select: {},
-      };
-      core.status.thisEventClickArea = [];
-    };
-
-    this.refreshBox = function () {
-      if (!core.status.event.id) return;
-      if (core.status.event.id == "toolbox") core.drawToolbox();
-      else core.drawEquipbox();
-    };
-
-    core.ui.closePanel = function () {
-      if (core.status.hero && core.status.hero.flags) {
-        // 清除全部临时变量
-        Object.keys(core.status.hero.flags).forEach(function (name) {
-          if (name.startsWith("@temp@") || /^arg\d+$/.test(name)) {
-            delete core.status.hero.flags[name];
-          }
-        });
-      }
-      this.clearUI();
-      core.maps.generateGroundPattern();
-      core.updateStatusBar(true);
-      core.unlockControl();
-      core.status.event.data = null;
-      core.status.event.id = null;
-      core.status.event.selection = null;
-      core.status.event.ui = null;
-      core.status.event.interval = null;
-      core.status.thisUIEventInfo = null;
-      core.status.thisEventClickArea = null;
-    };
-
-    this.getItemClsName = function (item) {
-      if (item == null) return itemClsName;
-      if (item.cls == "equips") {
-        if (typeof item.equip.type == "string") return item.equip.type;
-        var type = core.getEquipTypeById(item.id);
-        return core.status.globalAttribute.equipName[type];
-      } else return itemClsName[item.cls] || item.cls;
-    };
-
-    core.events.openToolbox = function (fromUserAction) {
-      if (core.isReplaying()) return;
-      if (!this._checkStatus("toolbox", fromUserAction)) return;
-      core.initThisEventInfo();
-      let info = core.status.thisUIEventInfo;
-      info.index = 1;
-      core.setIndexAndSelect("select");
-      core.drawToolbox();
-    };
-
-    core.events.openEquipbox = function (fromUserAction) {
-      if (core.isReplaying()) return;
-      if (!this._checkStatus("equipbox", fromUserAction)) return;
-      core.initThisEventInfo();
-      let info = core.status.thisUIEventInfo;
-      info.select.type = 0;
-      core.setIndexAndSelect("select");
-      core.drawEquipbox();
-    };
-
-    core.control._replay_toolbox = function () {
-      if (!core.isPlaying() || !core.isReplaying()) return;
-      if (!core.status.replay.pausing) return core.drawTip("请先暂停录像");
-      if (core.isMoving() || core.status.replay.animate || core.status.event.id)
-        return core.drawTip("请等待当前事件的处理结束");
-
-      core.lockControl();
-      core.status.event.id = "toolbox";
-      core.drawToolbox();
-    };
-
-    core.control._replay_equipbox = function () {
-      if (!core.isPlaying() || !core.isReplaying()) return;
-      if (!core.status.replay.pausing) return core.drawTip("请先暂停录像");
-      if (core.isMoving() || core.status.replay.animate || core.status.event.id)
-        return core.drawTip("请等待当前事件的处理结束");
-
-      core.lockControl();
-      core.status.event.id = "equipbox";
-      core.drawEquipbox();
-    };
-
-    core.control._replayAction_item = function (action) {
-      if (action.indexOf("item:") != 0) return false;
-      var itemId = action.substring(5);
-      if (!core.canUseItem(itemId)) return false;
-      if (
-        core.material.items[itemId].hideInReplay ||
-        core.status.replay.speed == 24
-      ) {
-        core.useItem(itemId, false, core.replay);
-        return true;
-      }
-      core.status.event.id = "toolbox";
-      core.initThisEventInfo();
-      var info = core.status.thisUIEventInfo;
-      var items = core.getToolboxItems("all");
-      core.setPageItems(1);
-      var index = items.indexOf(itemId) + 1;
-      info.page = Math.ceil(index / info.maxItem);
-      info.index = index % info.maxItem || info.maxItem;
-      core.setIndexAndSelect("select");
-      core.setPageItems(info.page);
-      core.drawToolbox();
-      setTimeout(function () {
-        core.ui.closePanel();
-        core.useItem(itemId, false, core.replay);
-      }, core.control.__replay_getTimeout());
-      return true;
-    };
-
-    core.control._replayAction_equip = function (action) {
-      if (action.indexOf("equip:") != 0) return false;
-      var itemId = action.substring(6);
-      var items = core.getToolboxItems("equips");
-      var index = items.indexOf(itemId) + 1;
-      if (index < 1) return false;
-      core.status.route.push(action);
-      if (
-        core.material.items[itemId].hideInReplay ||
-        core.status.replay.speed == 24
-      ) {
-        core.loadEquip(itemId, core.replay);
-        return true;
-      }
-      core.status.event.id = "equipbox";
-      core.initThisEventInfo();
-      var info = core.status.thisUIEventInfo;
-      core.setPageItems(1);
-      info.page = Math.ceil(index / info.maxItem);
-      info.index = index % info.maxItem || info.maxItem;
-      core.setIndexAndSelect("select");
-      core.setPageItems(info.page);
-      core.drawEquipbox();
-      setTimeout(function () {
-        core.ui.closePanel();
-        core.loadEquip(itemId, core.replay);
-      }, core.control.__replay_getTimeout());
-      return true;
-    };
-
-    core.control._replayAction_unEquip = function (action) {
-      if (action.indexOf("unEquip:") != 0) return false;
-      var equipType = parseInt(action.substring(8));
-      if (!core.isset(equipType)) return false;
-      core.status.route.push(action);
-      if (core.status.replay.speed == 24) {
-        core.unloadEquip(equipType, core.replay);
-        return true;
-      }
-      core.status.event.id = "equipbox";
-      core.initThisEventInfo();
-      var info = core.status.thisUIEventInfo;
-      core.setPageItems(1);
-      info.select.type = equipType;
-      core.setIndexAndSelect();
-      core.drawEquipbox();
-      setTimeout(function () {
-        core.ui.closePanel();
-        core.unloadEquip(equipType, core.replay);
-      }, core.control.__replay_getTimeout());
-      return true;
-    };
-
-    core.registerReplayAction("item", core.control._replayAction_item);
-    core.registerReplayAction("equip", core.control._replayAction_equip);
-    core.registerReplayAction("unEquip", core.control._replayAction_unEquip);
-  },
+		core.setAlpha(ctx, backgroundAlpha);
+		core.strokeRoundRect(
+			ctx,
+			x,
+			y,
+			w,
+			h,
+			borderRadius,
+			borderStyle,
+			borderWidth
+		);
+		core.fillRoundRect(
+			ctx,
+			start_x,
+			start_y,
+			width,
+			height,
+			borderRadius,
+			backgroundColor
+		);
+		core.setAlpha(ctx, 1);
+
+		///// *** 左栏配置
+		var leftbar_height = height;
+		// 左边栏宽度(width*0.6) 本身仅为坐标使用 需要与底下的rightbar_width(width*0.4)同时更改
+		var leftbar_width = width * 0.6;
+		///// ***
+
+		// xxx_right参数 代表最右侧坐标
+		var leftbar_right = start_x + leftbar_width - borderWidth / 2;
+		var leftbar_bottom = start_y + leftbar_height;
+		var leftbar_x = start_x;
+		var leftbar_y = start_y;
+
+		///// *** 道具栏配置
+		var boxName_color = "#fff";
+		var boxName_fontSize = 15;
+		var boxName_font = core.ui._buildFont(boxName_fontSize, true);
+		var arrow_x = 10 + start_x;
+		var arrow_y = 10 + start_y;
+		var arrow_width = 20;
+		var arrow_style = "white";
+		// 暂时只能是1 否则不太行 等待新样板(2.7.3)之后对drawArrow做优化
+		var arrow_lineWidth = 2;
+		// 右箭头
+		var rightArrow_right = leftbar_right - 10;
+		// 道具内栏顶部坐标 本质是通过该项 控制(道具栏顶部文字和箭头)与道具内栏顶部的间隔
+		var itembar_top = arrow_y + 15;
+		///// ***
+
+		var itembar_right = rightArrow_right;
+		var boxName =
+			core.status.event.id == "toolbox" ?
+			"\r[yellow]道具栏\r | 装备栏" :
+			"道具栏 | \r[yellow]装备栏\r";
+		core.drawArrow(
+			ctx,
+			arrow_x + arrow_width,
+			arrow_y,
+			arrow_x,
+			arrow_y,
+			arrow_style,
+			arrow_lineWidth
+		);
+		core.drawArrow(
+			ctx,
+			rightArrow_right - arrow_width,
+			arrow_y,
+			rightArrow_right,
+			arrow_y,
+			arrow_style,
+			arrow_lineWidth
+		);
+		core.setTextAlign(ctx, "center");
+		core.setTextBaseline(ctx, "middle");
+		var changeBox = function () {
+			var id = core.status.event.id;
+			core.closePanel();
+			if (id == "toolbox") core.openEquipbox();
+			else core.openToolbox();
+		};
+		core.fillText(
+			ctx,
+			boxName,
+			(leftbar_right + leftbar_x) / 2,
+			arrow_y + 2,
+			boxName_color,
+			boxName_font
+		);
+
+		///// *** 底栏按钮
+		var pageBtn_radius = 8;
+		// xxx_left 最左侧坐标
+		var pageBtn_left = leftbar_x + 3;
+		var pageBtn_right = leftbar_right - 3;
+		// xxx_bottom 最底部坐标
+		var pageBtn_bottom = leftbar_bottom - 2;
+		var pageBtn_borderStyle = "#fff";
+		var pageBtn_borderWidth = 2;
+		var pageText_color = "#fff";
+		// 底部按钮与上面的道具内栏的间隔大小
+		var bottomSpace = 8;
+		///// ***
+
+		drawItemListbox_setPageBtn(
+			ctx,
+			pageBtn_left,
+			pageBtn_right,
+			pageBtn_bottom,
+			pageBtn_radius,
+			pageBtn_borderStyle,
+			pageBtn_borderWidth
+		);
+		var page = info.page || 1;
+		var pageFontSize = pageBtn_radius * 2 - 4;
+		var pageFont = core.ui._buildFont(pageFontSize);
+		setPageItems(page);
+		var num = itemNum;
+		if (core.status.event.id == "equipbox") num -= 5;
+		var maxPage = info.maxPage;
+		var pageText = page + " / " + maxPage;
+		core.setTextAlign(ctx, "center");
+		core.setTextBaseline(ctx, "bottom");
+		core.fillText(
+			ctx,
+			pageText,
+			(leftbar_x + leftbar_right) / 2,
+			pageBtn_bottom,
+			pageText_color,
+			pageFont
+		);
+		addUIEventListener(
+			start_x,
+			start_y,
+			leftbar_right - start_x,
+			arrow_y - start_y + 13,
+			changeBox
+		);
+		var itembar_height = Math.ceil(
+			pageBtn_bottom -
+			pageBtn_radius * 2 -
+			pageBtn_borderWidth / 2 -
+			bottomSpace -
+			itembar_top
+		);
+		var oneItemHeight = (itembar_height - 4) / itemNum;
+		return {
+			x: start_x,
+			y: start_y,
+			width: width,
+			height: height,
+			leftbar_right: leftbar_right,
+			obj: {
+				x: arrow_x,
+				y: itembar_top,
+				width: itembar_right - arrow_x,
+				height: itembar_height,
+				oneItemHeight: oneItemHeight,
+			},
+		};
+	}
+
+	function drawItemListbox(ctx, obj) {
+		ctx = ctx || core.canvas.ui;
+		var itembar_x = obj.x,
+			itembar_y = obj.y,
+			itembar_width = obj.width,
+			itembar_height = obj.height,
+			itemNum = obj.itemNum,
+			oneItemHeight = obj.oneItemHeight;
+		var itembar_right = itembar_x + itembar_width;
+		var info = core.status.thisUIEventInfo || {};
+		var obj = {};
+		var page = info.page || 1,
+			index = info.index,
+			select = info.select || {};
+
+		///// *** 道具栏内栏配置
+		var itembar_style = "black";
+		var itembar_alpha = 0.7;
+		// 一个竖屏下减少道具显示的例子:
+		// if (core.domStyle.isVertical) itemNum = 10;
+		// 每个道具项的上下空隙占总高度的比例
+		var itembar_marginHeightRatio = 0.2;
+		// 左右间隔空隙
+		var item_marginLeft = 2;
+		var item_x = itembar_x + 2,
+			item_y = itembar_y + 2,
+			item_right = itembar_right - 2,
+			itemName_color = "#fff";
+		// 修改此项以更换闪烁光标
+		var item_selector = "winskin.webp";
+		///// ***
+
+		core.setAlpha(ctx, itembar_alpha);
+		core.fillRect(
+			ctx,
+			itembar_x,
+			itembar_y,
+			itembar_width,
+			itembar_height,
+			itembar_style
+		);
+		core.setAlpha(ctx, 1);
+		var pageItems = setPageItems(page);
+		var marginHeight = itembar_marginHeightRatio * oneItemHeight;
+		core.setTextBaseline(ctx, "middle");
+		var originColor = itemName_color;
+		for (var i = 0; i < pageItems.length; i++) {
+			itemName_color = originColor;
+			var item = pageItems[i];
+			// 设置某个的字体颜色的一个例子
+			// if (item.id == "xxx") itemName_color = "green";
+			drawItemListbox_drawItem(
+				ctx,
+				item_x,
+				item_right,
+				item_y,
+				oneItemHeight,
+				item_marginLeft,
+				marginHeight,
+				itemName_color,
+				pageItems[i]
+			);
+			if (index == i + 1)
+				core.ui._drawWindowSelector(
+					item_selector,
+					item_x + 1,
+					item_y - 1,
+					item_right - item_x - 2,
+					oneItemHeight - 2
+				);
+			item_y += oneItemHeight;
+		}
+	}
+
+	function drawToolboxRightbar(ctx, obj) {
+		ctx = ctx || core.canvas.ui;
+		var info = core.status.thisUIEventInfo || {};
+		var page = info.page || 1,
+			index = info.index || 1,
+			select = info.select || {};
+		var start_x = obj.x,
+			start_y = obj.y,
+			width = obj.width,
+			height = obj.height;
+		var toolboxRight = start_x + width,
+			toolboxBottom = start_y + height;
+
+		///// *** 侧边栏(rightbar)背景设置(物品介绍)
+		var rightbar_width = width * 0.4;
+		var rightbar_height = height;
+		var rightbar_lineWidth = 2;
+		var rightbar_lineStyle = "#fff";
+		///// ***
+
+		var rightbar_x = toolboxRight - rightbar_width - rightbar_lineWidth / 2;
+		var rightbar_y = start_y;
+		core.drawLine(
+			ctx,
+			rightbar_x,
+			rightbar_y,
+			rightbar_x,
+			rightbar_y + rightbar_height,
+			rightbar_lineStyle,
+			rightbar_lineWidth
+		);
+
+		// 获取道具id(有可能为null)
+		var itemId = select.id;
+		var item = core.material.items[itemId];
+
+		///// *** 侧边栏物品Icon信息
+		var iconRect_y = rightbar_y + 10;
+		// space:间距
+		// 这里布局设定iconRect与侧边栏左边框 itemName与工具栏右边框 itemRect与itemName的间距均为space
+		var space = 15;
+		var iconRect_x = rightbar_x + space;
+		var iconRect_radius = 2,
+			iconRect_width = 32,
+			iconRect_height = 32,
+			iconRect_style = "#fff",
+			iconRect_lineWidth = 2;
+		///// ***
+
+		var iconRect_bottom = iconRect_y + iconRect_height,
+			iconRect_right = iconRect_x + iconRect_width;
+
+		///// *** 侧边栏各项信息
+		var itemTextFontSize = 15,
+			itemText_x = iconRect_x - 4,
+			itemText_y = Math.floor(start_y + rightbar_height * 0.25), // 坐标取整防止模糊
+			itemClsFontSize = 15,
+			itemClsFont = core.ui._buildFont(itemClsFontSize),
+			itemClsColor = "#fff",
+			itemCls_x = itemText_x - itemClsFontSize / 2,
+			itemCls_middle = (iconRect_bottom + itemText_y) / 2, //_middle代表文字的中心y坐标
+			itemNameFontSize = 18,
+			itemNameColor = "#fff",
+			itemNameFont = core.ui._buildFont(itemNameFontSize, true);
+		var itemName_x = iconRect_right + space;
+		var itemName_middle =
+			iconRect_y + iconRect_height / 2 + iconRect_lineWidth;
+		// 修改这里可以编辑未选中道具时的默认值
+		var defaultItem = {
+			cls: "constants",
+			name: "未知道具",
+			text: "没有道具最永久",
+		};
+		var defaultEquip = {
+			cls: "equips",
+			name: "未知装备",
+			text: "一无所有,又何尝不是一种装备",
+			equip: {
+				type: "装备",
+			},
+		};
+		///// ***
+
+		var originItem = item;
+		if (core.status.event.id == "equipbox") item = item || defaultEquip;
+		item = item || defaultItem;
+		var itemCls = item.cls,
+			itemName = item.name,
+			itemText = item.text;
+		itemText = core.replaceText(itemText);
+		if (!itemText) itemText = "该道具无描述。";
+		/* 一个根据道具id修改道具名字(右栏)的例子
+		 * if (item.id == "xxx") itemNameColor = "red";
+		 */
+		var itemClsName = core.getItemClsName(item);
+		var itemNameMaxWidth =
+			rightbar_width - iconRect_width - iconRect_lineWidth * 2 - space * 2;
+		core.strokeRoundRect(
+			ctx,
+			iconRect_x,
+			iconRect_y,
+			iconRect_width,
+			iconRect_height,
+			iconRect_radius,
+			iconRect_style,
+			iconRect_lineWidth
+		);
+		if (item.id)
+			core.drawIcon(
+				ctx,
+				item.id,
+				iconRect_x + iconRect_lineWidth / 2,
+				iconRect_y + iconRect_lineWidth / 2,
+				iconRect_width - iconRect_lineWidth,
+				iconRect_height - iconRect_lineWidth
+			);
+		core.setTextAlign(ctx, "left");
+		core.setTextBaseline(ctx, "middle");
+		if (itemCls === "equips" && item.id) {
+			itemName = "【" + item.equipCls + "】" + itemName;
+		}
+		core.fillText(
+			ctx,
+			itemName,
+			itemName_x,
+			itemName_middle,
+			itemNameColor,
+			itemNameFont,
+			itemNameMaxWidth
+		);
+		if (!item.equip)
+			core.fillText(
+				ctx,
+				"【" + itemClsName + "】",
+				itemCls_x,
+				itemCls_middle,
+				itemClsColor,
+				itemClsFont
+			);
+
+		var statusText = "";
+		if (core.status.event.id == "equipbox") {
+			var type = item.equip.type;
+			if (typeof type == "string") type = core.getEquipTypeByName(type);
+			var compare = core.compareEquipment(item.id, core.getEquip(type));
+			var compare2;
+			if (item.equipCls === "双手剑")
+				compare2 = core.compareEquipment(null, core.getEquip(1));
+			if (
+				item.equipCls === "盾牌" &&
+				core.material.items[core.getEquip(0)]?.equipCls === "双手剑"
+			)
+				compare2 = core.compareEquipment(null, core.getEquip(0));
+			if (info.select.action == "unload")
+				compare = core.compareEquipment(null, item.id);
+			// --- 变化值...
+			for (var name in core.status.hero) {
+				if (typeof core.status.hero[name] != "number") continue;
+				var nowValue = core.getRealStatus(name);
+				// 查询新值
+				var newValue = Math.floor(
+					((core.getStatus(name) +
+							(compare.value[name] || 0) +
+							(compare2?.value[name] || 0)) *
+						(core.getBuff(name) * 100 +
+							(compare.percentage[name] || 0) +
+							(compare2?.percentage[name] || 0))) /
+					100
+				);
+				if (name === "mdef") {
+					var nowValue = core.getRealStatus(name);
+					var newValue = Math.round(
+						(core.getStatus(name) -
+							(compare.value[name] || 0) -
+							(compare2?.value[name] || 0)) *
+						(1 -
+							(1 - core.getBuff(name)) *
+							(compare.percentage[name] || 1) *
+							(compare2?.percentage[name] || 1))
+					);
+				}
+				if (nowValue == newValue) continue;
+				var color = newValue > nowValue ? "#00FF00" : "#FF0000";
+				nowValue = core.formatBigNumber(nowValue);
+				newValue = core.formatBigNumber(newValue);
+
+				if (name === "mdef") {
+					nowValue += "%";
+					newValue += "%";
+				}
+				statusText +=
+					core.getStatusLabel(name) +
+					" " +
+					nowValue +
+					"->\r[" +
+					color +
+					"]" +
+					newValue +
+					"\r\n";
+			}
+		}
+		itemText = statusText + itemText;
+		if (item.equip) {
+			core.drawTextContent(ctx, itemText, {
+				left: itemText_x,
+				top: itemCls_middle,
+				bold: false,
+				color: "white",
+				align: "left",
+				fontSize: itemTextFontSize,
+				maxWidth: rightbar_width -
+					(itemText_x - rightbar_x) * 2 +
+					itemTextFontSize / 2,
+			});
+		} else {
+			core.drawTextContent(ctx, itemText, {
+				left: itemText_x,
+				top: itemText_y,
+				bold: false,
+				color: "white",
+				align: "left",
+				fontSize: itemTextFontSize,
+				maxWidth: rightbar_width -
+					(itemText_x - rightbar_x) * 2 +
+					itemTextFontSize / 2,
+			});
+		}
+
+		///// *** 退出按钮设置
+		var btnRadius = 10;
+		var btnBorderWidth = 2;
+		var btnRight = toolboxRight - 2;
+		var btnBottom = toolboxBottom - 2;
+		var btnBorderStyle = "#fff";
+		///// ***
+
+		// 获取圆心位置
+		var btn_x = btnRight - btnRadius - btnBorderWidth / 2;
+		btn_y = btnBottom - btnRadius - btnBorderWidth / 2;
+		drawToolbox_setExitBtn(
+			ctx,
+			btn_x,
+			btn_y,
+			btnRadius,
+			btnBorderStyle,
+			btnBorderWidth
+		);
+
+		///// *** 使用按钮设置
+		var useBtnHeight = btnRadius * 2;
+		// 这里不设置useBtnWidth而是根据各项数据自动得出width
+		var useBtnRadius = useBtnHeight / 4;
+		var useBtn_x = rightbar_x + 4,
+			useBtn_y = btnBottom - useBtnHeight;
+		var useBtnBorderStyle = "#fff";
+		var useBtnBorderWidth = btnBorderWidth;
+		const batchUseBtn_x = useBtn_x + 50; // 个人觉得,搞这么多参数还不如硬编码
+		const hideBtn_y = useBtn_y - useBtnHeight - 8;
+		///// ***
+
+		drawToolbox_setUseBtn(
+			ctx,
+			useBtn_x,
+			useBtn_y,
+			useBtnRadius,
+			useBtnHeight,
+			useBtnBorderStyle,
+			useBtnBorderWidth
+		);
+		if (core.status.event.id === "toolbox") {
+			drawToolbox_setBatchUseBtn(
+				ctx,
+				batchUseBtn_x,
+				useBtn_y,
+				useBtnRadius,
+				useBtnHeight,
+				useBtnBorderStyle,
+				useBtnBorderWidth
+			);
+		}
+		drawToolbox_setHideBtn(
+			ctx,
+			useBtn_x,
+			hideBtn_y,
+			useBtnRadius,
+			useBtnHeight,
+			useBtnBorderStyle,
+			useBtnBorderWidth
+		);
+		drawToolbox_setShowHideBtn(
+			ctx,
+			rightbar_x,
+			useBtn_y,
+			useBtnHeight,
+			useBtnBorderStyle
+		);
+	}
+
+	function drawEquipbox_drawOthers(ctx, obj) {
+		var info = core.status.thisUIEventInfo;
+
+		///// *** 装备格设置
+		var equipList_lineWidth = 2;
+		var equipList_boxSize = 32;
+		var equipList_borderWidth = 2;
+		var equipList_borderStyle = "#fff";
+		var equipList_nameColor = "#fff";
+		///// ***
+
+		var equipList_x = obj.x + 4,
+			equipList_bottom = obj.obj.y - equipList_lineWidth,
+			equipList_y = equipList_bottom - obj.obj.oneItemHeight * reduceItem - 2,
+			equipList_height = equipList_bottom - equipList_y;
+		var equipList_right = obj.leftbar_right,
+			equipList_width = equipList_right - equipList_x;
+		core.drawLine(
+			ctx,
+			obj.x,
+			equipList_bottom + equipList_lineWidth / 2,
+			equipList_right,
+			equipList_bottom + equipList_lineWidth / 2,
+			equipList_borderStyle,
+			equipList_lineWidth
+		);
+		var toDrawList = core.status.globalAttribute.equipName,
+			len = toDrawList.length;
+
+		///// *** 装备格设置
+		var maxItem = 2;
+		var box_width = 32,
+			box_height = 32,
+			box_borderStyle = "#fff",
+			box_selectBorderStyle = "gold", // 选中的装备格的颜色
+			box_borderWidth = 2;
+		var boxName_fontSize = 14,
+			boxName_space = 2,
+			boxName_color = "#fff"; // 装备格名称与上面的装备格框的距离
+		var maxLine = Math.ceil(len / maxItem);
+		///// ***
+		var l = Math.sqrt(len);
+		if (Math.pow(l) == len && len != 4) {
+			if (l <= maxItem) maxItem = l;
+		}
+		maxItem = Math.min(toDrawList.length, maxItem);
+		info.equips = maxItem;
+
+		var boxName_font = core.ui._buildFont(boxName_fontSize);
+		// 总宽高减去所有装备格宽高得到空隙大小
+		var oneBoxWidth = box_width + box_borderWidth * 2;
+		var oneBoxHeight =
+			box_height + boxName_fontSize + boxName_space + 2 * box_borderWidth;
+		var space_y = (equipList_height - maxLine * oneBoxHeight) / (1 + maxLine),
+			space_x = (equipList_width - maxItem * oneBoxWidth) / (1 + maxItem);
+		var box_x = equipList_x + space_x,
+			box_y = equipList_y + space_y + 12;
+		for (var i = 0; i < 2; i++) {
+			var id = core.getEquip(i),
+				name = toDrawList[i];
+			if (i === 0) name = "主手";
+			if (i === 1) name = "副手";
+			var selectBorder = false;
+			if (core.status.thisUIEventInfo.select.type == i) selectBorder = true;
+			var borderStyle = selectBorder ?
+				box_selectBorderStyle :
+				box_borderStyle;
+			drawEquipbox_drawOne(
+				ctx,
+				name,
+				id,
+				box_x,
+				box_y,
+				box_width,
+				box_height,
+				boxName_space,
+				boxName_font,
+				boxName_color,
+				borderStyle,
+				box_borderWidth
+			);
+			var todo = new Function(
+				"core.clickOneEquipbox('" + id + "'," + i + ")"
+			);
+			addUIEventListener(
+				box_x - box_borderWidth / 2,
+				box_y - box_borderWidth / 2,
+				oneBoxWidth,
+				oneBoxHeight,
+				todo
+			);
+			box_x += space_x + oneBoxWidth;
+			if ((i + 1) % maxItem == 0) {
+				box_x = equipList_x + space_x;
+				box_y += space_y + oneBoxHeight;
+			}
+		}
+		if (core.material.items[core.getEquip(0)]?.equipCls === "双手剑") {
+			core.drawLine(
+				ctx,
+				equipList_x + space_x + space_x + oneBoxWidth,
+				equipList_y + space_y + 12,
+				equipList_x +
+				space_x +
+				space_x +
+				oneBoxWidth +
+				box_width +
+				box_borderWidth,
+				equipList_y + space_y + box_height + 12
+			);
+			core.drawLine(
+				ctx,
+				equipList_x + space_x + space_x + oneBoxWidth,
+				equipList_y + space_y + box_height + 12,
+				equipList_x +
+				space_x +
+				space_x +
+				oneBoxWidth +
+				box_width +
+				box_borderWidth,
+				equipList_y + space_y + 12
+			);
+		}
+		///// *** 装备格设置
+		var maxItem = 3;
+		var box_width = 32,
+			box_height = 32,
+			box_borderStyle = "#fff",
+			box_selectBorderStyle = "gold", // 选中的装备格的颜色
+			box_borderWidth = 2;
+		var boxName_fontSize = 14,
+			boxName_space = 2,
+			boxName_color = "#fff"; // 装备格名称与上面的装备格框的距离
+		var maxLine = Math.ceil(len / maxItem);
+		///// ***
+		var l = Math.sqrt(len);
+		if (Math.pow(l) == len && len != 4) {
+			if (l <= maxItem) maxItem = l;
+		}
+		maxItem = Math.min(toDrawList.length, maxItem);
+		info.equips = maxItem;
+
+		var boxName_font = core.ui._buildFont(boxName_fontSize);
+		// 总宽高减去所有装备格宽高得到空隙大小
+		var oneBoxWidth = box_width + box_borderWidth * 2;
+		var oneBoxHeight =
+			box_height + boxName_fontSize + boxName_space + 2 * box_borderWidth;
+		var space_y = (equipList_height - maxLine * oneBoxHeight) / (1 + maxLine),
+			space_x = (equipList_width - maxItem * oneBoxWidth) / (1 + maxItem);
+		var box_x = equipList_x + space_x,
+			box_y = equipList_y + space_y + space_y + oneBoxHeight;
+		for (var i = 2; i < len; i++) {
+			var id = core.getEquip(i),
+				name = toDrawList[i];
+			var selectBorder = false;
+			if (core.status.thisUIEventInfo.select.type == i) selectBorder = true;
+			var borderStyle = selectBorder ?
+				box_selectBorderStyle :
+				box_borderStyle;
+			drawEquipbox_drawOne(
+				ctx,
+				name,
+				id,
+				box_x,
+				box_y,
+				box_width,
+				box_height,
+				boxName_space,
+				boxName_font,
+				boxName_color,
+				borderStyle,
+				box_borderWidth
+			);
+			var todo = new Function(
+				"core.clickOneEquipbox('" + id + "'," + i + ")"
+			);
+			addUIEventListener(
+				box_x - box_borderWidth / 2,
+				box_y - box_borderWidth / 2,
+				oneBoxWidth,
+				oneBoxHeight,
+				todo
+			);
+			box_x += space_x + oneBoxWidth;
+		}
+	}
+
+	this.drawToolbox = function (ctx) {
+		ctx = ctx || core.canvas.ui;
+		core.status.thisEventClickArea = [];
+
+		var info = drawBoxBackground(ctx);
+		info.itemNum = itemNum;
+		drawItemListbox(ctx, info.obj);
+		drawToolboxRightbar(ctx, info);
+		core.setTextBaseline(ctx, "alphabetic");
+		core.setTextAlign("left");
+	};
+
+	var reduceItem = 4;
+	this.drawEquipbox = function (ctx) {
+		ctx = ctx || core.canvas.ui;
+		core.status.thisEventClickArea = [];
+		var info = drawBoxBackground(ctx);
+		info.itemNum = itemNum - reduceItem;
+		info.obj.y += info.obj.oneItemHeight * reduceItem;
+		info.obj.height -= info.obj.oneItemHeight * reduceItem;
+		drawItemListbox(ctx, info.obj);
+		drawEquipbox_drawOthers(ctx, info);
+		drawToolboxRightbar(ctx, info);
+		core.setTextBaseline(ctx, "alphabetic");
+		core.setTextAlign("left");
+	};
+
+	function drawEquipbox_drawOne(
+		ctx,
+		name,
+		id,
+		x,
+		y,
+		width,
+		height,
+		space,
+		font,
+		color,
+		style,
+		lineWidth
+	) {
+		if (id)
+			core.drawIcon(
+				ctx,
+				id,
+				x + lineWidth / 2,
+				y + lineWidth / 2,
+				width,
+				height
+			);
+		core.strokeRect(
+			ctx,
+			x,
+			y,
+			width + lineWidth,
+			height + lineWidth,
+			style,
+			lineWidth
+		);
+		core.setTextAlign(ctx, "center");
+		core.setTextBaseline(ctx, "top");
+		var tx = (x + x + lineWidth / 2 + width) / 2,
+			ty = y + height + (lineWidth / 2) * 3 + space;
+		core.fillText(ctx, name, tx, ty, color, font);
+
+		core.setAlpha(ctx, 1);
+
+		core.setTextBaseline(ctx, "alphabetic");
+		core.setTextAlign("left");
+	}
+
+	function drawItemListbox_drawItem(
+		ctx,
+		left,
+		right,
+		top,
+		height,
+		marginLeft,
+		marginHeight,
+		style,
+		id
+	) {
+		var info = core.status.thisUIEventInfo;
+		var nowClick = info.index;
+		var item = core.material.items[id] || {};
+		var name = item.name || "???";
+		var num = core.itemCount(id) || 0;
+		var fontSize = Math.floor(height - marginHeight * 2);
+		core.setTextAlign(ctx, "right");
+		var numText = "x" + num;
+		core.fillText(
+			ctx,
+			numText,
+			right - marginLeft,
+			top + height / 2,
+			style,
+			core.ui._buildFont(fontSize)
+		);
+
+		const hideInfo = core.getFlag("hideInfo", {});
+		if (
+			item &&
+			(hideInfo.hasOwnProperty(id) ? hideInfo[id] : item.hideInToolbox)
+		)
+			core.setAlpha(ctx, 0.5);
+
+		if (name != "???")
+			core.drawIcon(
+				ctx,
+				id,
+				left + marginLeft,
+				top + marginHeight,
+				fontSize,
+				fontSize
+			);
+		var text_x = left + marginLeft + fontSize + 2;
+		var maxWidth = right - core.calWidth(ctx, numText) - text_x;
+		core.setTextAlign(ctx, "left");
+		core.fillText(
+			ctx,
+			name,
+			text_x,
+			top + height / 2,
+			style,
+			core.ui._buildFont(fontSize),
+			maxWidth
+		);
+		core.setAlpha(ctx, 1);
+
+		var todo = new Function("core.clickItemFunc('" + id + "');");
+		addUIEventListener(left, top, right - left, height, todo);
+	}
+
+	function setPageItems(page) {
+		var num = itemNum;
+		if (core.status.event.id == "equipbox") num -= reduceItem;
+		var info = core.status.thisUIEventInfo;
+		if (!info) return;
+		page = page || info.page;
+		var items = core.getToolboxItems(
+			core.status.event.id == "toolbox" ? "all" : "equips",
+			core.getFlag("showHideItem", false)
+		);
+		info.allItems = items;
+		var maxPage = Math.ceil(items.length / num);
+		info.maxPage = maxPage;
+		var pageItems = items.slice((page - 1) * num, page * num);
+		info.pageItems = pageItems;
+		info.maxItem = pageItems.length;
+		if (items.length == 0 && pageItems.length == 0) info.index = null;
+		if (pageItems.length == 0 && info.page > 1) {
+			info.page = Math.max(1, info.page - 1);
+			return setPageItems(info.page);
+		}
+		return pageItems;
+	}
+
+	function drawToolbox_setExitBtn(ctx, x, y, r, style, lineWidth) {
+		core.strokeCircle(ctx, x, y, r, style, lineWidth);
+		ctx.textAlign = "center";
+		ctx.textBaseline = "middle";
+		var textSize = Math.sqrt(2) * r;
+		core.fillText(
+			ctx,
+			"x",
+			x,
+			y,
+			style,
+			core.ui._buildFont(textSize),
+			textSize
+		);
+		core.setTextAlign(ctx, "start");
+		core.setTextBaseline(ctx, "top");
+
+		var todo = function () {
+			core.closePanel();
+		};
+		addUIEventListener(x - r, y - r, r * 2, r * 2, todo);
+	}
+
+	function drawToolbox_setUseBtn(ctx, x, y, r, h, style, lineWidth) {
+		core.setTextAlign(ctx, "left");
+		core.setTextBaseline(ctx, "top");
+		var fontSize = h - 4;
+		var font = core.ui._buildFont(fontSize);
+		var text = core.status.event.id == "toolbox" ? "使用" : "装备";
+		if (core.status.thisUIEventInfo.select.action == "unload") text = "卸下";
+		var w = core.calWidth(ctx, text, font) + 2 * r + lineWidth / 2;
+
+		core.strokeRoundRect(ctx, x, y, w, h, r, style, lineWidth);
+		core.fillText(ctx, text, x + r, y + lineWidth / 2 + 2, style, font);
+
+		var todo = function () {
+			core.useSelectItemInBox();
+		};
+		addUIEventListener(x, y, w, h, todo);
+	}
+
+	function getSelectedItem() {
+		var info = core.status.thisUIEventInfo;
+		if (
+			!(
+				info && ["toolbox", "equipbox"].includes(core.status.event.id)
+			)
+		) {
+			core.drawFailTip("发生了未知错误!");
+			return;
+		}
+		return info.select.id;
+	}
+
+	function batchUse(item, count) {
+		try {
+			const itemCount = core.itemCount(item);
+			if (count > itemCount) count = itemCount;
+			core.closePanel();
+			for (let i = 0; i < count; i++) {
+				if (core.canUseItem(item)) core.useItem(item);
+				else return;
+			}
+		} catch (e) {
+			console.error(e);
+			core.drawFailTip("批量使用时出现未知错误!");
+		}
+	}
+
+	function drawToolbox_setBatchUseBtn(ctx, x, y, r, h, style, lineWidth) {
+		try {
+			const selectedItem = getSelectedItem();
+			let canBatchUse = eval(core.material.items[selectedItem]?.canBatchUse);
+			if (!canBatchUse) return;
+		} catch (error) {
+			console.error(error);
+			return;
+		}
+		core.setTextAlign(ctx, "left");
+		core.setTextBaseline(ctx, "top");
+		var fontSize = h - 4;
+		var font = core.ui._buildFont(fontSize);
+		var text = "批量使用";
+		var w = core.calWidth(ctx, text, font) + 2 * r + lineWidth / 2;
+
+		core.strokeRoundRect(ctx, x, y, w, h, r, style, lineWidth);
+		core.fillText(ctx, text, x + r, y + lineWidth / 2 + 2, style, font);
+
+		var todo = function () {
+			core.utils.myprompt("输入要使用该物品的次数(0~99)。", null, (value) => {
+				value = parseInt(value);
+				const id = getSelectedItem();
+
+				if (Number.isNaN(value) || value < 0 || value > 99) {
+					core.drawFailTip("输入不合法!");
+					return;
+				}
+				if (!core.canUseItem(id)) {
+					core.drawFailTip("当前无法使用该道具!");
+					return;
+				}
+				core.closePanel();
+				batchUse(id, value);
+			});
+		};
+		addUIEventListener(x, y, w, h, todo);
+	}
+
+	function drawToolbox_setHideBtn(ctx, x, y, r, h, style, lineWidth) {
+		core.setTextAlign(ctx, "left");
+		core.setTextBaseline(ctx, "top");
+		var fontSize = h - 4;
+		var font = core.ui._buildFont(fontSize);
+		var text = "显示/隐藏";
+		var w = core.calWidth(ctx, text, font) + 2 * r + lineWidth / 2;
+
+		core.strokeRoundRect(ctx, x, y, w, h, r, style, lineWidth);
+		core.fillText(ctx, text, x + r, y + lineWidth / 2 + 2, style, font);
+
+		var todo = function () {
+			//debugger;
+			var id = getSelectedItem();
+			let hideInfo = core.getFlag("hideInfo", {});
+			console.log(id);
+			if (hideInfo.hasOwnProperty(id)) {
+				hideInfo[id] = !hideInfo[id];
+				core.setFlag("hideInfo", hideInfo);
+			} else {
+				hideInfo[id] = !core.material.items[id].hideInToolbox;
+				core.setFlag("hideInfo", hideInfo);
+			}
+			if (core.status.event.id === "toolbox") core.plugin.drawToolbox();
+			else if (core.status.event.id === "equipbox")
+				core.plugin.drawEquipbox();
+		};
+		addUIEventListener(x, y, w, h, todo);
+	}
+
+	ui.prototype.getToolboxItems = function (cls, showHide) {
+		let list = Object.keys(core.status.hero.items[cls] || {});
+		if (cls === "all") {
+			for (let name in core.status.hero.items) {
+				if (name == "equips") continue;
+				list = list.concat(Object.keys(core.status.hero.items[name]));
+			}
+			if (!showHide)
+				list = list.filter(function (id2) {
+					const hideInfo = core.getFlag("hideInfo", {});
+					if (hideInfo.hasOwnProperty(id2)) return !hideInfo[id2];
+					else return !core.material.items[id2].hideInToolbox;
+				});
+			list = list.sort();
+			return list;
+		}
+		if (cls === "equips") {
+			if (!showHide)
+				list = list.filter(function (id2) {
+					const hideInfo = core.getFlag("hideInfo", {});
+					if (hideInfo.hasOwnProperty(id2)) return !hideInfo[id2];
+					else return !core.material.items[id2].hideInToolbox;
+				});
+
+			list = list.sort();
+			return list;
+		}
+		if (this.uidata.getToolboxItems) {
+			return this.uidata.getToolboxItems(cls, showHide);
+		}
+		if (!showHide)
+			list = list.filter(function (id2) {
+				return !core.material.items[id2].hideInToolbox;
+			});
+		list = list.sort();
+		return list;
+	};
+
+	function drawToolbox_setShowHideBtn(ctx, x, y, h, style) {
+		core.setTextAlign(ctx, "left");
+		core.setTextBaseline(ctx, "top");
+		var fontSize = h - 6;
+		var font = core.ui._buildFont(fontSize);
+		var text = "显示隐藏";
+		var w = core.calWidth(ctx, text, font);
+		h += 4;
+		const squareSize = h - 6;
+
+		x -= w + squareSize + 26;
+
+		const border = 2;
+		core.fillRect(ctx, x, y, squareSize, squareSize, " #F5F5F5");
+		if (core.hasFlag("showHideItem")) {
+			core.fillRect(
+				ctx,
+				x + border,
+				y + border,
+				squareSize - 2 * border,
+				squareSize - 2 * border,
+				"lime"
+			);
+		}
+		core.fillText(ctx, text, x + squareSize + 2, y + 4, style, font);
+
+		var todo = function () {
+			core.setFlag("showHideItem", !core.getFlag("showHideItem", false));
+			if (core.status.event.id === "toolbox") core.plugin.drawToolbox();
+			else if (core.status.event.id === "equipbox")
+				core.plugin.drawEquipbox();
+		};
+		addUIEventListener(x, y, w, h, todo);
+	}
+
+	function drawItemListbox_setPageBtn(
+		ctx,
+		left,
+		right,
+		bottom,
+		r,
+		style,
+		lineWidth
+	) {
+		var offset = lineWidth / 2 + r;
+
+		var x = left + offset;
+		var y = bottom - offset;
+		var pos = (Math.sqrt(2) / 2) * (r - lineWidth / 2);
+		core.fillPolygon(
+			ctx,
+			[
+				[x - pos, y],
+				[x + pos - 2, y - pos],
+				[x + pos - 2, y + pos],
+			],
+			style
+		);
+		core.strokeCircle(ctx, x, y, r, style, lineWidth);
+		var todo = function () {
+			core.addItemListboxPage(-1);
+		};
+		addUIEventListener(x - r - 2, y - r - 2, r * 2 + 4, r * 2 + 4, todo);
+
+		x = right - offset;
+		core.fillPolygon(
+			ctx,
+			[
+				[x + pos, y],
+				[x - pos + 2, y - pos],
+				[x - pos + 2, y + pos],
+			],
+			style
+		);
+		core.strokeCircle(ctx, x, y, r, style, lineWidth);
+		var todo = function () {
+			core.addItemListboxPage(1);
+		};
+		addUIEventListener(x - r - 2, y - r - 2, r * 2 + 4, r * 2 + 4, todo);
+	}
+
+	this.clickItemFunc = function (id) {
+		var info = core.status.thisUIEventInfo;
+		if (!info) return;
+		if (info.select.id == id) return core.useSelectItemInBox();
+		info.select = {};
+		info.select.id = id;
+		core.setIndexAndSelect("index");
+		refreshBox();
+	};
+
+	this.clickOneEquipbox = function (id, type) {
+		var info = core.status.thisUIEventInfo;
+		if (!info) return;
+		if (info.select.id == id && info.select.type == type)
+			core.useSelectItemInBox();
+		else
+			core.status.thisUIEventInfo.select = {
+				id: id,
+				type: type,
+				action: "unload",
+			};
+		return refreshBox();
+	};
+
+	this.useSelectItemInBox = function () {
+		var info = core.status.thisUIEventInfo;
+		if (!info) return;
+		if (!info.select.id) return;
+		var id = info.select.id;
+		if (core.status.event.id == "toolbox") {
+			core.events.tryUseItem(id);
+			// core.closePanel();
+		} else if (core.status.event.id == "equipbox") {
+			var action = info.select.action || "load";
+			info.index = 1;
+			if (action == "load") {
+				var type = core.getEquipTypeById(id);
+				let equipClsid = core.material.items[id]?.equipCls;
+				let equipCls0 = core.material.items[core.getEquip(0)]?.equipCls;
+				let equipCls1 = core.material.items[core.getEquip(1)]?.equipCls;
+				if (equipClsid === "双手剑") {
+					core.unloadEquip(0, function () {
+						core.status.route.push("unEquip:" + 0);
+					});
+					core.unloadEquip(1, function () {
+						core.status.route.push("unEquip:" + 1);
+					});
+				}
+				if (
+					equipCls0 === "双手剑" &&
+					!(equipClsid === "饰品" || equipClsid === "护具")
+				) {
+					core.unloadEquip(0, function () {
+						core.status.route.push("unEquip:" + 0);
+					});
+				}
+				core.loadEquip(id, function () {
+					core.status.route.push("equip:" + id);
+					info.select.type = type;
+					core.setIndexAndSelect("select");
+					core.drawEquipbox();
+				});
+			} else {
+				var type = info.select.type;
+				core.unloadEquip(type, function () {
+					core.status.route.push("unEquip:" + type);
+					info.select.type = type;
+					info.select.action = "load";
+					core.setIndexAndSelect("select");
+					core.drawEquipbox();
+				});
+			}
+		}
+		core.updateStatusBar();
+	};
+
+	this.setIndexAndSelect = function (toChange) {
+		var info = core.status.thisUIEventInfo;
+		if (!info) return;
+		setPageItems(info.page);
+		var index = info.index || 1;
+		var items = info.pageItems;
+
+		info.select.action = null;
+		info.select.type = null;
+		if (toChange == "index") info.index = items.indexOf(info.select.id) + 1;
+		info.select.id = items[info.index - 1];
+	};
+
+	this.addItemListboxPage = function (num) {
+		var info = core.status.thisUIEventInfo;
+		if (!info) return;
+		var maxPage = info.maxPage || 1;
+		info.page = info.page || 1;
+		info.page += num;
+		if (info.page <= 0) info.page = maxPage;
+		if (info.page > maxPage) info.page = 1;
+		info.index = 1;
+		setPageItems(info.page);
+		core.setIndexAndSelect("select");
+		refreshBox();
+	};
+
+	this.addItemListboxIndex = function (num) {
+		var info = core.status.thisUIEventInfo;
+		if (!info) return;
+		var maxItem = info.maxItem || 0;
+		info.index = info.index || 0;
+		info.index += num;
+		if (info.index <= 0) info.index = 1;
+		if (info.index > maxItem) info.index = maxItem;
+		core.setIndexAndSelect("select");
+		refreshBox();
+	};
+
+	this.addEquipboxType = function (num) {
+		var info = core.status.thisUIEventInfo;
+		var type = info.select.type;
+		if (type == null && num > 0) info.select.type = 0;
+		else info.select.type = type + num;
+		var max = core.status.globalAttribute.equipName.length;
+		if (info.select.type >= max) {
+			info.select = {};
+			core.setIndexAndSelect("select");
+			return core.addItemListboxPage(0);
+		} else {
+			var m = Math.abs(info.select.type);
+			if (info.select.type < 0) info.select.type = max - m;
+			core.setIndexAndSelect("select");
+			refreshBox();
+			return;
+		}
+	};
+
+	core.actions._keyDownToolbox = function (keycode) {
+		if (!core.status.thisEventClickArea) return;
+		if (keycode == 37) {
+			// left
+			core.addItemListboxPage(-1);
+			return;
+		}
+		if (keycode == 38) {
+			// up
+			core.addItemListboxIndex(-1);
+			return;
+		}
+		if (keycode == 39) {
+			// right
+			core.addItemListboxPage(1);
+			return;
+		}
+		if (keycode == 40) {
+			// down
+			core.addItemListboxIndex(1);
+			return;
+		}
+	};
+
+	////// 工具栏界面时,放开某个键的操作 //////
+	core.actions._keyUpToolbox = function (keycode) {
+		if (keycode == 81) {
+			core.ui.closePanel();
+			if (core.isReplaying()) core.control._replay_equipbox();
+			else core.openEquipbox();
+			return;
+		}
+		if (keycode == 84 || keycode == 27 || keycode == 88) {
+			core.closePanel();
+			return;
+		}
+		if (keycode == 13 || keycode == 32 || keycode == 67) {
+			var info = core.status.thisUIEventInfo;
+			if (info.select) {
+				core.useSelectItemInBox();
+			}
+			return;
+		}
+	};
+
+	core.actions._keyDownEquipbox = function (keycode) {
+		if (!core.status.thisEventClickArea) return;
+		if (keycode == 37) {
+			// left
+			var info = core.status.thisUIEventInfo;
+			if (info.index != null) return core.addItemListboxPage(-1);
+			return core.addEquipboxType(-1);
+		}
+		if (keycode == 38) {
+			// up
+			var info = core.status.thisUIEventInfo;
+			if (info.index == 1) {
+				info.select.type = core.status.globalAttribute.equipName.length - 1;
+				core.setIndexAndSelect();
+				return refreshBox();
+			}
+			if (info.index) return core.addItemListboxIndex(-1);
+			return core.addEquipboxType(-1 * info.equips);
+		}
+		if (keycode == 39) {
+			// right
+			var info = core.status.thisUIEventInfo;
+			if (info.index != null) return core.addItemListboxPage(1);
+			return core.addEquipboxType(1);
+		}
+		if (keycode == 40) {
+			// down
+			var info = core.status.thisUIEventInfo;
+			if (info.index) return core.addItemListboxIndex(1);
+			return core.addEquipboxType(info.equips);
+		}
+	};
+
+	core.actions._keyUpEquipbox = function (keycode, altKey) {
+		if (altKey && keycode >= 48 && keycode <= 57) {
+			core.items.quickSaveEquip(keycode - 48);
+			return;
+		}
+		if (keycode == 84) {
+			core.ui.closePanel();
+			if (core.isReplaying()) core.control._replay_toolbox();
+			else core.openToolbox();
+			return;
+		}
+		if (keycode == 81 || keycode == 27 || keycode == 88) {
+			core.closePanel();
+			return;
+		}
+		if (keycode == 13 || keycode == 32 || keycode == 67) {
+			var info = core.status.thisUIEventInfo;
+			if (info.select) core.useSelectItemInBox();
+			return;
+		}
+	};
+
+	core.registerAction(
+		"ondown",
+		"inEventClickAction",
+		function (x, y, px, py) {
+			if (!core.status.thisEventClickArea) return false;
+			var info = core.status.thisEventClickArea;
+			for (var i = 0; i < info.length; i++) {
+				var obj = info[i];
+				if (
+					px >= obj.x &&
+					px <= obj.x + obj.width &&
+					py > obj.y &&
+					py < obj.y + obj.height
+				) {
+					if (obj.todo) obj.todo();
+					break;
+				}
+			}
+			return true;
+		},
+		51
+	);
+	core.registerAction(
+		"onclick",
+		"stopClick",
+		function () {
+			if (core.status.thisEventClickArea) return true;
+		},
+		51
+	);
+
+	function addUIEventListener(x, y, width, height, todo) {
+		if (!core.status.thisEventClickArea) return;
+		var obj = {
+			x: x,
+			y: y,
+			width: width,
+			height: height,
+			todo: todo,
+		};
+		core.status.thisEventClickArea.push(obj);
+	}
+
+	this.initThisEventInfo = function () {
+		core.status.thisUIEventInfo = {
+			page: 1,
+			select: {},
+		};
+		core.status.thisEventClickArea = [];
+	};
+
+	function refreshBox() {
+		if (!core.status.event.id) return;
+		if (core.status.event.id == "toolbox") core.drawToolbox();
+		else core.drawEquipbox();
+	}
+
+	core.ui.closePanel = function () {
+		core.dom.saveLoad.style.display = "none"
+		if (core.status.hero && core.status.hero.flags) {
+			// 清除全部临时变量
+			Object.keys(core.status.hero.flags).forEach(function (name) {
+				if (name.startsWith("@temp@") || /^arg\d+$/.test(name)) {
+					delete core.status.hero.flags[name];
+				}
+			});
+		}
+		this.clearUI();
+		core.maps.generateGroundPattern();
+		core.updateStatusBar(true);
+		core.unlockControl();
+		core.status.event.data = null;
+		core.status.event.id = null;
+		core.status.event.selection = null;
+		core.status.event.ui = null;
+		core.status.event.interval = null;
+		core.status.thisUIEventInfo = null;
+		core.status.thisEventClickArea = null;
+	};
+
+	this.getItemClsName = function (item) {
+		if (item == null) return itemClsName;
+		if (item.cls == "equips") {
+			if (typeof item.equip.type == "string") return item.equip.type;
+			var type = core.getEquipTypeById(item.id);
+			return core.status.globalAttribute.equipName[type];
+		} else return itemClsName[item.cls] || item.cls;
+	};
+
+	core.events.openToolbox = function (fromUserAction) {
+		if (core.isReplaying()) return;
+		if (!this._checkStatus("toolbox", fromUserAction)) return;
+		core.initThisEventInfo();
+		let info = core.status.thisUIEventInfo;
+		info.index = 1;
+		core.setIndexAndSelect("select");
+		core.drawToolbox();
+	};
+
+	core.events.openEquipbox = function (fromUserAction) {
+		if (core.isReplaying()) return;
+		if (!this._checkStatus("equipbox", fromUserAction)) return;
+		core.initThisEventInfo();
+		let info = core.status.thisUIEventInfo;
+		info.select.type = 0;
+		core.setIndexAndSelect("select");
+		core.drawEquipbox();
+	};
+
+	core.control._replay_toolbox = function () {
+		if (!core.isPlaying() || !core.isReplaying()) return;
+		if (!core.status.replay.pausing) return core.drawTip("请先暂停录像");
+		if (core.isMoving() || core.status.replay.animate || core.status.event.id)
+			return core.drawTip("请等待当前事件的处理结束");
+
+		core.lockControl();
+		core.status.event.id = "toolbox";
+		core.drawToolbox();
+	};
+
+	core.control._replay_equipbox = function () {
+		if (!core.isPlaying() || !core.isReplaying()) return;
+		if (!core.status.replay.pausing) return core.drawTip("请先暂停录像");
+		if (core.isMoving() || core.status.replay.animate || core.status.event.id)
+			return core.drawTip("请等待当前事件的处理结束");
+
+		core.lockControl();
+		core.status.event.id = "equipbox";
+		core.drawEquipbox();
+	};
+
+	core.control._replayAction_item = function (action) {
+		if (action.indexOf("item:") != 0) return false;
+		var itemId = action.substring(5);
+		if (!core.canUseItem(itemId)) return false;
+		if (
+			core.material.items[itemId].hideInReplay ||
+			core.status.replay.speed == 24
+		) {
+			core.useItem(itemId, false, core.replay);
+			return true;
+		}
+		core.status.event.id = "toolbox";
+		core.initThisEventInfo();
+		var info = core.status.thisUIEventInfo;
+		var items = core.getToolboxItems(
+			"all",
+			core.getFlag("showHideItem", false)
+		);
+		setPageItems(1);
+		var index = items.indexOf(itemId) + 1;
+		info.page = Math.ceil(index / info.maxItem);
+		info.index = index % info.maxItem || info.maxItem;
+		core.setIndexAndSelect("select");
+		setPageItems(info.page);
+		core.drawToolbox();
+		setTimeout(function () {
+			core.ui.closePanel();
+			core.useItem(itemId, false, core.replay);
+		}, core.control.__replay_getTimeout());
+		return true;
+	};
+
+	core.control._replayAction_equip = function (action) {
+		if (action.indexOf("equip:") != 0) return false;
+		var itemId = action.substring(6);
+		var items = core.getToolboxItems(
+			"equips",
+			core.getFlag("showHideItem", false)
+		);
+		var index = items.indexOf(itemId) + 1;
+		if (index < 1) {
+			core.removeFlag("__doNotCheckAutoEvents__");
+			return false;
+		}
+
+		var cb = function () {
+			var next = core.status.replay.toReplay[0] || "";
+			if (!next.startsWith("equip:") && !next.startsWith("unEquip:")) {
+				core.removeFlag("__doNotCheckAutoEvents__");
+				core.checkAutoEvents();
+			}
+			core.replay();
+		};
+		core.setFlag("__doNotCheckAutoEvents__", true);
+
+		core.status.route.push(action);
+		if (
+			core.material.items[itemId].hideInReplay ||
+			core.status.replay.speed == 24
+		) {
+			core.loadEquip(itemId, cb);
+			return true;
+		}
+		core.status.event.id = "equipbox";
+		core.initThisEventInfo();
+		var info = core.status.thisUIEventInfo;
+		setPageItems(1);
+		info.page = Math.ceil(index / info.maxItem);
+		info.index = index % info.maxItem || info.maxItem;
+		core.setIndexAndSelect("select");
+		setPageItems(info.page);
+		core.drawEquipbox();
+		setTimeout(function () {
+			core.ui.closePanel();
+			core.loadEquip(itemId, cb);
+		}, core.control.__replay_getTimeout());
+		return true;
+	};
+
+	core.control._replayAction_unEquip = function (action) {
+		if (action.indexOf("unEquip:") != 0) return false;
+		var equipType = parseInt(action.substring(8));
+		if (!core.isset(equipType)) {
+			core.removeFlag("__doNotCheckAutoEvents__");
+			return false;
+		}
+
+		var cb = function () {
+			var next = core.status.replay.toReplay[0] || "";
+			if (!next.startsWith("equip:") && !next.startsWith("unEquip:")) {
+				core.removeFlag("__doNotCheckAutoEvents__");
+				core.checkAutoEvents();
+			}
+			core.replay();
+		};
+		core.setFlag("__doNotCheckAutoEvents__", true);
+
+		core.status.route.push(action);
+		if (core.status.replay.speed == 24) {
+			core.unloadEquip(equipType, cb);
+			return true;
+		}
+		core.status.event.id = "equipbox";
+		core.initThisEventInfo();
+		var info = core.status.thisUIEventInfo;
+		setPageItems(1);
+		info.select.type = equipType;
+		core.setIndexAndSelect();
+		core.drawEquipbox();
+		setTimeout(function () {
+			core.ui.closePanel();
+			core.unloadEquip(equipType, cb);
+		}, core.control.__replay_getTimeout());
+		return true;
+	};
+	core.registerReplayAction("item", core.control._replayAction_item);
+	core.registerReplayAction("equip", core.control._replayAction_equip);
+	core.registerReplayAction("unEquip", core.control._replayAction_unEquip);
+},
     "技能树": function () {
     // 在此增加新插件
     //
@@ -8034,7 +9068,42 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
         return arr;
       }
     }
+    /**
+     * 图片叠加滤镜
+     * @param {img} image 需要叠加的图片,
+     * @param {color} style 需要叠加的颜色
+     */
+    function imagelighter(image, style) {
+      // 创建一个canvas元素
+      const canvas = document.createElement("canvas");
+      const ctx = canvas.getContext("2d");
 
+      // 设置canvas的尺寸与图片相同
+      canvas.width = image.width;
+      canvas.height = image.height;
+      ctx.drawImage(image, 0, 0);
+      // 创建一个临时canvas用于红色滤镜
+      const tempCanvas = document.createElement("canvas");
+      const tempCtx = tempCanvas.getContext("2d");
+      tempCanvas.width = image.width;
+      tempCanvas.height = image.height;
+
+      // 在临时canvas上绘制红色滤镜
+      tempCtx.fillStyle = style ?? "rgba(255, 0, 0, 0.5)"; // 半透明红色
+      tempCtx.fillRect(0, 0, tempCanvas.width, tempCanvas.height);
+      // 使用lighter混合模式叠加红色滤镜
+      ctx.globalCompositeOperation = "lighter";
+      ctx.drawImage(tempCanvas, 0, 0);
+      // 使用destination-in混合模式保留原始图片的透明度
+      ctx.globalCompositeOperation = "destination-in";
+      ctx.drawImage(image, 0, 0);
+
+      // 恢复默认混合模式
+      ctx.globalCompositeOperation = "source-over";
+
+      // 返回处理后的canvas
+      return canvas;
+    }
     /**
      * 获取一个方向的反方向
      * @example backDir('up'); // 'down'
@@ -8388,6 +9457,35 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
       yellowgreen: "#9acd32",
       transparent: "#0000",
     };
+    // 计算两个数的最大公约数
+    function gcdOfTwo(a, b) {
+      while (b !== 0) {
+        let temp = b;
+        b = a % b;
+        a = temp;
+      }
+      return a;
+    }
+
+    // 计算任意项整数的最大公约数
+    function gcd(...numbers) {
+      if (numbers.length < 2) {
+        throw new Error("至少需要两个数");
+      }
+      return numbers.reduce((a, b) => gcdOfTwo(a, b));
+    }
+    // 计算两个数的最小公倍数
+    function lcmOfTwo(a, b) {
+      return (a * b) / gcdOfTwo(a, b);
+    }
+
+    // 计算任意项整数的最小公倍数
+    function lcm(...numbers) {
+      if (numbers.length < 2) {
+        throw new Error("至少需要两个数");
+      }
+      return numbers.reduce((a, b) => lcmOfTwo(a, b));
+    }
 
     if (has(core.plugin.utils)) {
       throw new ReferenceError(
@@ -8395,6 +9493,11 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
       );
     }
     core.plugin.utils = {
+      imagelighter,
+      gcdOfTwo,
+      lcmOfTwo,
+      gcd,
+      lcm,
       has,
       slide,
       backDir,
@@ -8449,8 +9552,8 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
     };
   },
     "音频系统": function () {
-	// 在此增加新插件
-	/*首先,在造塔群下载所需的库文件,然后放置在塔目录下的 libs/thirdparty 或其他目录下,之后在 index.html 的最后加上下面这几行:
+    // 在此增加新插件
+    /*首先,在造塔群下载所需的库文件,然后放置在塔目录下的 libs/thirdparty 或其他目录下,之后在 index.html 的最后加上下面这几行:
 	
 	<script src="libs/thirdparty/ogg-vorbis-decoder.min.js"></script>
 
@@ -8458,2357 +9561,2099 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
 
 	<script src="libs/thirdparty/codec-parser.min.js"></script>
 	*/
-	// 将__enable置为false将关闭插件
-	let __enable = true;
-	if (!__enable || main.mode === "editor") return;
-	const { OggOpusDecoderWebWorker } = window["ogg-opus-decoder"];
-	const { OggVorbisDecoderWebWorker } = window["ogg-vorbis-decoder"];
-	const { CodecParser } = window.CodecParser;
-	const { Transition, linear } = core.plugin.animate;
-
-	const audio = new Audio();
-	const AudioStatus = {
-		Playing: 0,
-		Pausing: 1,
-		Paused: 2,
-		Stoping: 3,
-		Stoped: 4,
-	};
-	const supportMap = new Map();
-	const AudioType = {
-		Mp3: "audio/mpeg",
-		Wav: 'audio/wav; codecs="1"',
-		Flac: "audio/flac",
-		Opus: 'audio/ogg; codecs="opus"',
-		Ogg: 'audio/ogg; codecs="vorbis"',
-		Aac: "audio/aac",
-	};
-	/**
-	 * 检查一种音频类型是否能被播放
-	 * @param type 音频类型 AudioType
-	 */
-	function isAudioSupport(type) {
-		if (supportMap.has(type)) return supportMap.get(type);
-		else {
-			const support = audio.canPlayType(type);
-			const canPlay = support === "maybe" || support === "probably";
-			supportMap.set(type, canPlay);
-			return canPlay;
-		}
-	}
-
-	const typeMap = new Map([
-		["ogg", AudioType.Ogg],
-		["mp3", AudioType.Mp3],
-		["wav", AudioType.Wav],
-		["flac", AudioType.Flac],
-		["opus", AudioType.Opus],
-		["aac", AudioType.Aac],
-	]);
-
-	/**
-	 * 根据文件名拓展猜测其类型
-	 * @param file 文件名 string
-	 */
-	function guessTypeByExt(file) {
-		const ext = /\.[a-zA-Z\d]+$/.exec(file);
-		if (!ext?.[0]) return "";
-		const type = ext[0].slice(1);
-		return typeMap.get(type.toLocaleLowerCase()) ?? "";
-	}
-
-	isAudioSupport(AudioType.Ogg);
-	isAudioSupport(AudioType.Mp3);
-	isAudioSupport(AudioType.Wav);
-	isAudioSupport(AudioType.Flac);
-	isAudioSupport(AudioType.Opus);
-	isAudioSupport(AudioType.Aac);
-
-	function isNil(value) {
-		return value === void 0 || value === null;
-	}
-
-	function sleep(time) {
-		return new Promise((res) => setTimeout(res, time));
-	}
-	class AudioEffect {
-		constructor(ac) {}
-		/**
-		 * 连接至其他效果器
-		 * @param target 目标输入 IAudioInput
-		 * @param output 当前效果器输出通道 Number
-		 * @param input 目标效果器的输入通道 Number
-		 */
-		connect(target, output, input) {
-			this.output.connect(target.input, output, input);
-		}
-
-		/**
-		 * 与其他效果器取消连接
-		 * @param target 目标输入 IAudioInput
-		 * @param output 当前效果器输出通道 Number
-		 * @param input 目标效果器的输入通道 Number
-		 */
-		disconnect(target, output, input) {
-			if (!target) {
-				if (!isNil(output)) {
-					this.output.disconnect(output);
-				} else {
-					this.output.disconnect();
-				}
-			} else {
-				if (!isNil(output)) {
-					if (!isNil(input)) {
-						this.output.disconnect(target.input, output, input);
-					} else {
-						this.output.disconnect(target.input, output);
-					}
-				} else {
-					this.output.disconnect(target.input);
-				}
-			}
-		}
-	}
-
-	class StereoEffect extends AudioEffect {
-		constructor(ac) {
-			super(ac);
-			const panner = ac.createPanner();
-			this.input = panner;
-			this.output = panner;
-		}
-
-		/**
-		 * 设置音频朝向,x正方形水平向右,y正方形垂直于地面向上,z正方向垂直屏幕远离用户
-		 * @param x 朝向x坐标 Number
-		 * @param y 朝向y坐标 Number
-		 * @param z 朝向z坐标 Number
-		 */
-		setOrientation(x, y, z) {
-			this.output.orientationX.value = x;
-			this.output.orientationY.value = y;
-			this.output.orientationZ.value = z;
-		}
-		/**
-		 * 设置音频位置,x正方形水平向右,y正方形垂直于地面向上,z正方向垂直屏幕远离用户
-		 * @param x 位置x坐标 Number
-		 * @param y 位置y坐标 Number
-		 * @param z 位置z坐标 Number
-		 */
-		setPosition(x, y, z) {
-			this.output.positionX.value = x;
-			this.output.positionY.value = y;
-			this.output.positionZ.value = z;
-		}
-		end() {}
-
-		start() {}
-	}
-	class VolumeEffect extends AudioEffect {
-		constructor(ac) {
-			super(ac);
-			const gain = ac.createGain();
-			this.input = gain;
-			this.output = gain;
-		}
-
-		/**
-		 * 设置音量大小
-		 * @param volume 音量大小 Number
-		 */
-		setVolume(volume) {
-			this.output.gain.value = volume;
-		}
-
-		/**
-		 * 获取音量大小 Number
-		 */
-		getVolume() {
-			return this.output.gain.value;
-		}
-
-		end() {}
-
-		start() {}
-	}
-	class ChannelVolumeEffect extends AudioEffect {
-		/** 所有的音量控制节点 */
-
-		constructor(ac) {
-			super(ac);
-			/** 所有的音量控制节点 */
-			this.gain = [];
-			const splitter = ac.createChannelSplitter();
-			const merger = ac.createChannelMerger();
-			this.output = merger;
-			this.input = splitter;
-			for (let i = 0; i < 6; i++) {
-				const gain = ac.createGain();
-				splitter.connect(gain, i);
-				gain.connect(merger, 0, i);
-				this.gain.push(gain);
-			}
-		}
-
-		/**
-		 * 设置某个声道的音量大小
-		 * @param channel 要设置的声道,可填0-5 Number
-		 * @param volume 这个声道的音量大小 Number
-		 */
-		setVolume(channel, volume) {
-			if (!this.gain[channel]) return;
-			this.gain[channel].gain.value = volume;
-		}
-
-		/**
-		 * 获取某个声道的音量大小,可填0-5
-		 * @param channel 要获取的声道 Number
-		 */
-		getVolume(channel) {
-			if (!this.gain[channel]) return 0;
-			return this.gain[channel].gain.value;
-		}
-
-		end() {}
-
-		start() {}
-	}
-	class DelayEffect extends AudioEffect {
-		constructor(ac) {
-			super(ac);
-
-			const delay = ac.createDelay();
-			this.input = delay;
-			this.output = delay;
-		}
-
-		/**
-		 * 设置延迟时长
-		 * @param delay 延迟时长,单位秒 Number
-		 */
-		setDelay(delay) {
-			this.output.delayTime.value = delay;
-		}
-
-		/**
-		 * 获取延迟时长
-		 */
-		getDelay() {
-			return this.output.delayTime.value;
-		}
-
-		end() {}
-
-		start() {}
-	}
-	class EchoEffect extends AudioEffect {
-		constructor(ac) {
-			super(ac);
-			/** 当前增益 */
-			this.gain = 0.5;
-			/** 是否正在播放 */
-			this.playing = false;
-			const delay = ac.createDelay();
-			const gain = ac.createGain();
-			gain.gain.value = 0.5;
-			delay.delayTime.value = 0.05;
-			delay.connect(gain);
-			gain.connect(delay);
-			/** 延迟节点 */
-			this.delay = delay;
-			/** 反馈增益节点 */
-			this.gainNode = gain;
-
-			this.input = gain;
-			this.output = gain;
-		}
-
-		/**
-		 * 设置回声反馈增益大小
-		 * @param gain 增益大小,范围 0-1,大于等于1的视为0.5,小于0的视为0 Number
-		 */
-		setFeedbackGain(gain) {
-			const resolved = gain >= 1 ? 0.5 : gain < 0 ? 0 : gain;
-			this.gain = resolved;
-			if (this.playing) this.gainNode.gain.value = resolved;
-		}
-
-		/**
-		 * 设置回声间隔时长
-		 * @param delay 回声时长,范围 0.01-Infinity,小于0.01的视为0.01 Number
-		 */
-		setEchoDelay(delay) {
-			const resolved = delay < 0.01 ? 0.01 : delay;
-			this.delay.delayTime.value = resolved;
-		}
-
-		/**
-		 * 获取反馈节点增益
-		 */
-		getFeedbackGain() {
-			return this.gain;
-		}
-
-		/**
-		 * 获取回声间隔时长
-		 */
-		getEchoDelay() {
-			return this.delay.delayTime.value;
-		}
-
-		end() {
-			this.playing = false;
-			const echoTime = Math.ceil(Math.log(0.001) / Math.log(this.gain)) + 10;
-			sleep(this.delay.delayTime.value * echoTime).then(() => {
-				if (!this.playing) this.gainNode.gain.value = 0;
-			});
-		}
-
-		start() {
-			this.playing = true;
-			this.gainNode.gain.value = this.gain;
-		}
-	}
-
-	class StreamLoader {
-		constructor(url) {
-			/** 传输目标  Set<IStreamReader>*/
-			this.target = new Set();
-			this.loading = false;
-		}
-
-		/**
-		 * 将加载流传递给字节流读取对象
-		 * @param reader 字节流读取对象 IStreamReader
-		 */
-		pipe(reader) {
-			if (this.loading) {
-				console.warn(
-					"Cannot pipe new StreamReader object when stream is loading."
-				);
-				return;
-			}
-			this.target.add(reader);
-			reader.piped(this);
-			return this;
-		}
-
-		async start() {
-			if (this.loading) return;
-			this.loading = true;
-			const response = await window.fetch(this.url);
-			const stream = response.body;
-			if (!stream) {
-				console.error("Cannot get reader when fetching '" + this.url + "'.");
-				return;
-			}
-			// 获取读取器
-			/** 读取流对象 */
-			this.stream = stream;
-			const reader = response.body?.getReader();
-			const targets = [...this.target];
-
-			await Promise.all(targets.map((v) => v.start(stream, this, response)));
-			if (reader && reader.read) {
-				// 开始流传输
-				while (true) {
-					const { value, done } = await reader.read();
-					await Promise.all(
-						targets.map((v) => v.pump(value, done, response))
-					);
-					if (done) break;
-				}
-			} else {
-				// 如果不支持流传输
-				const buffer = await response.arrayBuffer();
-				const data = new Uint8Array(buffer);
-				await Promise.all(targets.map((v) => v.pump(data, true, response)));
-			}
-			// 开始流传输
-			while (true) {
-				const { value, done } = await reader.read();
-				await Promise.all(targets.map((v) => v.pump(value, done, response)));
-				if (done) break;
-			}
-
-			this.loading = false;
-			targets.forEach((v) => v.end(true));
-
-			//
-		}
-
-		cancel(reason) {
-			if (!this.stream) return;
-			this.stream.cancel(reason);
-			this.loading = false;
-			this.target.forEach((v) => v.end(false, reason));
-		}
-	}
-	const fileSignatures = [
-		[AudioType.Mp3, [0x49, 0x44, 0x33]],
-		[AudioType.Ogg, [0x4f, 0x67, 0x67, 0x53]],
-		[AudioType.Wav, [0x52, 0x49, 0x46, 0x46]],
-		[AudioType.Flac, [0x66, 0x4c, 0x61, 0x43]],
-		[AudioType.Aac, [0xff, 0xf1]],
-		[AudioType.Aac, [0xff, 0xf9]],
-	];
-	const oggHeaders = [
-		[AudioType.Opus, [0x4f, 0x70, 0x75, 0x73, 0x48, 0x65, 0x61, 0x64]],
-	];
-
-	function checkAudioType(data) {
-		let audioType = "";
-		// 检查头文件获取音频类型,仅检查前256个字节
-		const toCheck = data.slice(0, 256);
-		for (const [type, value] of fileSignatures) {
-			if (value.every((v, i) => toCheck[i] === v)) {
-				audioType = type;
-				break;
-			}
-		}
-		if (audioType === AudioType.Ogg) {
-			// 如果是ogg的话,进一步判断是不是opus
-			for (const [key, value] of oggHeaders) {
-				const has = toCheck.some((_, i) => {
-					return value.every((v, ii) => toCheck[i + ii] === v);
-				});
-				if (has) {
-					audioType = key;
-					break;
-				}
-			}
-		}
-
-		return audioType;
-	}
-	class AudioDecoder {
-		/**
-		 * 注册一个解码器
-		 * @param type 要注册的解码器允许解码的类型
-		 * @param decoder 解码器对象
-		 */
-		static registerDecoder(type, decoder) {
-			if (!this.decoderMap) this.decoderMap = new Map();
-			if (this.decoderMap.has(type)) {
-				console.warn(
-					"Audio stream decoder for audio type '" +
-					type +
-					"' has already existed."
-				);
-				return;
-			}
-
-			this.decoderMap.set(type, decoder);
-		}
-
-		/**
-		 * 解码音频数据
-		 * @param data 音频文件数据
-		 * @param player AudioPlayer实例
-		 */
-		static async decodeAudioData(data, player) {
-			// 检查头文件获取音频类型,仅检查前256个字节
-			const toCheck = data.slice(0, 256);
-			const type = checkAudioType(data);
-			if (type === "") {
-				console.error(
-					"Unknown audio type. Header: '" + [...toCheck]
-					.map((v) => v.toString().padStart(2, "0"))
-					.join(" ")
-					.toUpperCase() +
-					"'"
-				);
-				return null;
-			}
-			if (isAudioSupport(type)) {
-				if (data.buffer instanceof ArrayBuffer) {
-					return player.ac.decodeAudioData(data.buffer);
-				} else {
-					return null;
-				}
-			} else {
-				const Decoder = this.decoderMap.get(type);
-				if (!Decoder) {
-					return null;
-				} else {
-					const decoder = new Decoder();
-					await decoder.create();
-					const decodedData = await decoder.decode(data);
-					if (!decodedData) return null;
-					const buffer = player.ac.createBuffer(
-						decodedData.channelData.length,
-						decodedData.channelData[0].length,
-						decodedData.sampleRate
-					);
-					decodedData.channelData.forEach((v, i) => {
-						buffer.copyToChannel(v, i);
-					});
-					decoder.destroy();
-					return buffer;
-				}
-			}
-		}
-	}
-
-	class VorbisDecoder {
-		/**
-		 * 创建音频解码器
-		 */
-		async create() {
-			this.decoder = new OggVorbisDecoderWebWorker();
-			await this.decoder.ready;
-		}
-		/**
-		 * 摧毁这个解码器
-		 */
-		destroy() {
-			this.decoder?.free();
-		}
-		/**
-		 * 解码流数据
-		 * @param data 流数据
-		 */
-
-		async decode(data) {
-			return this.decoder?.decode(data);
-		}
-		/**
-		 * 解码整个文件
-		 * @param data 文件数据
-		 */
-		async decodeAll(data) {
-			return this.decoder?.decodeFile(data);
-		}
-		/**
-		 * 当音频解码完成后,会调用此函数,需要返回之前还未解析或未返回的音频数据。调用后,该解码器将不会被再次使用
-		 */
-		async flush() {
-			return this.decoder?.flush();
-		}
-	}
-
-	class OpusDecoder {
-		/**
-		 * 创建音频解码器
-		 */
-		async create() {
-			this.decoder = new OggOpusDecoderWebWorker();
-			await this.decoder.ready;
-		}
-		/**
-		 * 摧毁这个解码器
-		 */
-		destroy() {
-			this.decoder?.free();
-		}
-		/**
-		 * 解码流数据
-		 * @param data 流数据
-		 */
-		async decode(data) {
-			return this.decoder?.decode(data);
-		}
-		/**
-		 * 解码整个文件
-		 * @param data 文件数据
-		 */
-		async decodeAll(data) {
-			return this.decoder?.decodeFile(data);
-		}
-		/**
-		 * 当音频解码完成后,会调用此函数,需要返回之前还未解析或未返回的音频数据。调用后,该解码器将不会被再次使用
-		 */
-		async flush() {
-			return await this.decoder?.flush();
-		}
-	}
-	const mimeTypeMap = {
-		[AudioType.Aac]: "audio/aac",
-		[AudioType.Flac]: "audio/flac",
-		[AudioType.Mp3]: "audio/mpeg",
-		[AudioType.Ogg]: "application/ogg",
-		[AudioType.Opus]: "application/ogg",
-		[AudioType.Wav]: "application/ogg",
-	};
-
-	function isOggPage(data) {
-		return !isNil(data.isFirstPage);
-	}
-	class AudioStreamSource {
-		constructor(context) {
-			this.output = context.createBufferSource();
-			/** 是否已经完全加载完毕 */
-			this.loaded = false;
-			/** 是否正在播放 */
-			this.playing = false;
-			/** 已经缓冲了多长时间,如果缓冲完那么跟歌曲时长一致 */
-			this.buffered = 0;
-			/** 已经缓冲的采样点数量 */
-			this.bufferedSamples = 0;
-			/** 歌曲时长,加载完毕之前保持为 0 */
-			this.duration = 0;
-			/** 在流传输阶段,至少缓冲多长时间的音频之后才开始播放,单位秒 */
-			this.bufferPlayDuration = 1;
-			/** 音频的采样率,未成功解析出之前保持为 0 */
-			this.sampleRate = 0;
-			//是否循环播放
-			this.loop = false;
-			/** 上一次播放是从何时开始的 */
-			this.lastStartWhen = 0;
-			/** 开始播放时刻 */
-			this.lastStartTime = 0;
-			/** 上一次播放的缓存长度 */
-			this.lastBufferSamples = 0;
-
-			/** 是否已经获取到头文件 */
-			this.headerRecieved = false;
-			/** 音频类型 */
-			this.audioType = "";
-			/** 每多长时间组成一个缓存 Float32Array */
-			this.bufferChunkSize = 10;
-			/** 缓存音频数据,每 bufferChunkSize 秒钟组成一个 Float32Array,用于流式解码 */
-			this.audioData = [];
-
-			this.errored = false;
-			this.ac = context;
-		}
-		/** 当前已经播放了多长时间 */
-		get currentTime() {
-			return this.ac.currentTime - this.lastStartTime + this.lastStartWhen;
-		}
-		/**
-		 * 设置每个缓存数据的大小,默认为10秒钟一个缓存数据
-		 * @param size 每个缓存数据的时长,单位秒
-		 */
-		setChunkSize(size) {
-			if (this.controller?.loading || this.loaded) return;
-			this.bufferChunkSize = size;
-		}
-		on(event, fn, context) {}
-		piped(controller) {
-			this.controller = controller;
-		}
-
-		async pump(data, done) {
-			if (!data || this.errored) return;
-			if (!this.headerRecieved) {
-				// 检查头文件获取音频类型,仅检查前256个字节
-				const toCheck = data.slice(0, 256);
-				this.audioType = checkAudioType(data);
-				if (!this.audioType) {
-					console.error(
-						"Unknown audio type. Header: '" + [...toCheck]
-						.map((v) => v.toString(16).padStart(2, "0"))
-						.join(" ")
-						.toUpperCase() +
-						"'"
-					);
-					return;
-				}
-				// 创建解码器
-				const Decoder = AudioDecoder.decoderMap.get(this.audioType);
-				if (!Decoder) {
-					this.errored = true;
-					console.error(
-						"Cannot decode stream source type of '" +
-						this.audioType +
-						"', since there is no registered decoder for that type."
-					);
-					return Promise.reject(
-						`Cannot decode stream source type of '${this.audioType}', since there is no registered decoder for that type.`
-					);
-				}
-				this.decoder = new Decoder();
-				// 创建数据解析器
-				const mime = mimeTypeMap[this.audioType];
-				const parser = new CodecParser(mime);
-				this.parser = parser;
-				await this.decoder.create();
-				this.headerRecieved = true;
-			}
-
-			const decoder = this.decoder;
-			const parser = this.parser;
-			if (!decoder || !parser) {
-				this.errored = true;
-				return Promise.reject(
-					"No parser or decoder attached in this AudioStreamSource"
-				);
-			}
-
-			await this.decodeData(data, decoder, parser);
-			if (done) await this.decodeFlushData(decoder, parser);
-			this.checkBufferedPlay();
-		}
-
-		/**
-		 * 检查采样率,如果还未解析出采样率,那么将设置采样率,如果当前采样率与之前不同,那么发出警告
-		 */
-		checkSampleRate(info) {
-			for (const one of info) {
-				const frame = isOggPage(one) ? one.codecFrames[0] : one;
-				if (frame) {
-					const rate = frame.header.sampleRate;
-					if (this.sampleRate === 0) {
-						this.sampleRate = rate;
-						break;
-					} else {
-						if (rate !== this.sampleRate) {
-							console.warn("Sample rate in stream audio must be constant.");
-						}
-					}
-				}
-			}
-		}
-
-		/**
-		 * 解析音频数据
-		 */
-		async decodeData(data, decoder, parser) {
-			// 解析音频数据
-			const audioData = await decoder.decode(data);
-			if (!audioData) return;
-			// @ts-expect-error 库类型声明错误
-			const audioInfo = [...parser.parseChunk(data)];
-
-			// 检查采样率
-			this.checkSampleRate(audioInfo);
-			// 追加音频数据
-			this.appendDecodedData(audioData, audioInfo);
-		}
-
-		/**
-		 * 解码剩余数据
-		 */
-		async decodeFlushData(decoder, parser) {
-			const audioData = await decoder.flush();
-			if (!audioData) return;
-			// @ts-expect-error 库类型声明错误
-			const audioInfo = [...parser.flush()];
-
-			this.checkSampleRate(audioInfo);
-			this.appendDecodedData(audioData, audioInfo);
-		}
-
-		/**
-		 * 追加音频数据
-		 */
-		appendDecodedData(data, info) {
-			const channels = data.channelData.length;
-			if (channels === 0) return;
-			if (this.audioData.length !== channels) {
-				this.audioData = [];
-				for (let i = 0; i < channels; i++) {
-					this.audioData.push([]);
-				}
-			}
-			// 计算出应该放在哪
-			const chunk = this.sampleRate * this.bufferChunkSize;
-			const sampled = this.bufferedSamples;
-			const pushIndex = Math.floor(sampled / chunk);
-			const bufferIndex = sampled % chunk;
-			const dataLength = data.channelData[0].length;
-			let buffered = 0;
-			let nowIndex = pushIndex;
-			let toBuffer = bufferIndex;
-			while (buffered < dataLength) {
-				const rest = toBuffer !== 0 ? chunk - bufferIndex : chunk;
-
-				for (let i = 0; i < channels; i++) {
-					const audioData = this.audioData[i];
-					if (!audioData[nowIndex]) {
-						audioData.push(new Float32Array(chunk));
-					}
-					const toPush = data.channelData[i].slice(buffered, buffered + rest);
-
-					audioData[nowIndex].set(toPush, toBuffer);
-				}
-				buffered += rest;
-				nowIndex++;
-				toBuffer = 0;
-			}
-
-			this.buffered +=
-				info.reduce((prev, curr) => prev + curr.duration, 0) / 1000;
-			this.bufferedSamples += info.reduce(
-				(prev, curr) => prev + curr.samples,
-				0
-			);
-		}
-
-		/**
-		 * 检查已缓冲内容,并在未开始播放时播放
-		 */
-		checkBufferedPlay() {
-			if (this.playing || this.sampleRate === 0) return;
-			const played = this.lastBufferSamples / this.sampleRate;
-			const dt = this.buffered - played;
-			if (this.loaded) {
-				this.playAudio(played);
-				return;
-			}
-			if (dt < this.bufferPlayDuration) return;
-
-			this.lastBufferSamples = this.bufferedSamples;
-			// 需要播放
-			this.mergeBuffers();
-			if (!this.buffer) return;
-			if (this.playing) this.output.stop();
-			this.createSourceNode(this.buffer);
-			this.output.loop = false;
-			this.output.start(0, played);
-			this.lastStartTime = this.ac.currentTime;
-			this.playing = true;
-			this.output.addEventListener("ended", () => {
-				this.playing = false;
-				this.checkBufferedPlay();
-			});
-		}
-
-		mergeBuffers() {
-			const buffer = this.ac.createBuffer(
-				this.audioData.length,
-				this.bufferedSamples,
-				this.sampleRate
-			);
-			const chunk = this.sampleRate * this.bufferChunkSize;
-			const bufferedChunks = Math.floor(this.bufferedSamples / chunk);
-			const restLength = this.bufferedSamples % chunk;
-			for (let i = 0; i < this.audioData.length; i++) {
-				const audio = this.audioData[i];
-				const data = new Float32Array(this.bufferedSamples);
-				for (let j = 0; j < bufferedChunks; j++) {
-					data.set(audio[j], chunk * j);
-				}
-				if (restLength !== 0) {
-					data.set(
-						audio[bufferedChunks].slice(0, restLength),
-						chunk * bufferedChunks
-					);
-				}
-
-				buffer.copyToChannel(data, i, 0);
-			}
-			this.buffer = buffer;
-		}
-
-		async start() {
-			delete this.buffer;
-			this.headerRecieved = false;
-			this.audioType = "";
-			this.errored = false;
-			this.buffered = 0;
-			this.sampleRate = 0;
-			this.bufferedSamples = 0;
-			this.duration = 0;
-			this.loaded = false;
-			if (this.playing) this.output.stop();
-			this.playing = false;
-			this.lastStartTime = this.ac.currentTime;
-		}
-
-		end(done, reason) {
-			if (done && this.buffer) {
-				this.loaded = true;
-				delete this.controller;
-				this.mergeBuffers();
-
-				this.duration = this.buffered;
-				this.audioData = [];
-				this.decoder?.destroy();
-				delete this.decoder;
-				delete this.parser;
-			} else {
-				console.warn(
-					"Unexpected end when loading stream audio, reason: '" +
-					(reason ?? "") +
-					"'"
-				);
-			}
-		}
-
-		playAudio(when) {
-			if (!this.buffer) return;
-			this.lastStartTime = this.ac.currentTime;
-			if (this.playing) this.output.stop();
-			if (this.player.status !== AudioStatus.Playing) {
-				this.player.status = AudioStatus.Playing;
-			}
-			this.createSourceNode(this.buffer);
-			this.output.start(0, when);
-			this.playing = true;
-
-			this.output.addEventListener("ended", () => {
-				this.playing = false;
-				if (this.player.status === AudioStatus.Playing) {
-					this.player.status = AudioStatus.Stoped;
-				}
-				if (this.loop && !this.output.loop) this.play(0);
-			});
-		}
-		/**
-		 * 开始播放这个音频源
-		 */
-		play(when) {
-			if (this.playing || this.errored) return;
-			if (this.loaded && this.buffer) {
-				this.playing = true;
-				this.playAudio(when);
-			} else {
-				this.controller?.start();
-			}
-		}
-
-		createSourceNode(buffer) {
-			if (!this.target) return;
-			const node = this.ac.createBufferSource();
-			node.buffer = buffer;
-			if (this.playing) this.output.stop();
-			this.playing = false;
-			this.output = node;
-			node.connect(this.target.input);
-			node.loop = this.loop;
-		}
-		/**
-		 * 停止播放这个音频源
-		 * @returns 音频暂停的时刻 number
-		 */
-		stop() {
-			if (this.playing) this.output.stop();
-			this.playing = false;
-			return this.ac.currentTime - this.lastStartTime;
-		}
-		/**
-		 * 连接到音频路由图上,每次调用播放的时候都会执行一次
-		 * @param target 连接至的目标 IAudioInput
-		 */
-		connect(target) {
-			this.target = target;
-		}
-		/**
-		 * 设置是否循环播放
-		 * @param loop 是否循环 boolean)
-		 */
-		setLoop(loop) {
-			this.loop = loop;
-		}
-	}
-	class AudioElementSource {
-		constructor(context) {
-			const audio = new Audio();
-			audio.preload = "none";
-			this.output = context.createMediaElementSource(audio);
-			this.audio = audio;
-			this.ac = context;
-			audio.addEventListener("play", () => {
-				this.playing = true;
-				if (this.player.status !== AudioStatus.Playing) {
-					this.player.status = AudioStatus.Playing;
-				}
-			});
-			audio.addEventListener("ended", () => {
-				this.playing = false;
-				if (this.player.status === AudioStatus.Playing) {
-					this.player.status = AudioStatus.Stoped;
-				}
-			});
-		}
-		get duration() {
-			return this.audio.duration;
-		}
-		get currentTime() {
-			return this.audio.currentTime;
-		}
-		/**
-		 * 设置音频源的路径
-		 * @param url 音频路径
-		 */
-		setSource(url) {
-			this.audio.src = url;
-		}
-
-		play(when = 0) {
-			if (this.playing) return;
-			this.audio.currentTime = when;
-			this.audio.play();
-		}
-
-		stop() {
-			this.audio.pause();
-			this.playing = false;
-			if (this.player.status === AudioStatus.Playing) {
-				this.player.status = AudioStatus.Stoped;
-			}
-			return this.audio.currentTime;
-		}
-
-		connect(target) {
-			this.output.connect(target.input);
-		}
-
-		setLoop(loop) {
-			this.audio.loop = loop;
-		}
-	}
-	class AudioBufferSource {
-		constructor(context) {
-			this.output = context.createBufferSource();
-			/** 是否循环 */
-			this.loop = false;
-			/** 上一次播放是从何时开始的 */
-			this.lastStartWhen = 0;
-			/** 播放开始时刻 */
-			this.lastStartTime = 0;
-			this.duration = 0;
-			this.ac = context;
-		}
-		get currentTime() {
-			return this.ac.currentTime - this.lastStartTime + this.lastStartWhen;
-		}
-
-		/**
-		 * 设置音频源数据
-		 * @param buffer 音频源,可以是未解析的 ArrayBuffer,也可以是已解析的 AudioBuffer
-		 */
-		async setBuffer(buffer) {
-			if (buffer instanceof ArrayBuffer) {
-				this.buffer = await this.ac.decodeAudioData(buffer);
-			} else {
-				this.buffer = buffer;
-			}
-			this.duration = this.buffer.duration;
-		}
-
-		play(when) {
-			if (this.playing || !this.buffer) return;
-			this.playing = true;
-			this.lastStartTime = this.ac.currentTime;
-			if (this.player.status !== AudioStatus.Playing) {
-				this.player.status = AudioStatus.Playing;
-			}
-			this.createSourceNode(this.buffer);
-			this.output.start(0, when);
-			this.output.addEventListener("ended", () => {
-				this.playing = false;
-				if (this.player.status === AudioStatus.Playing) {
-					this.player.status = AudioStatus.Stoped;
-				}
-				if (this.loop && !this.output.loop) this.play(0);
-			});
-		}
-
-		createSourceNode(buffer) {
-			if (!this.target) return;
-			const node = this.ac.createBufferSource();
-			node.buffer = buffer;
-			this.output = node;
-			node.connect(this.target.input);
-			node.loop = this.loop;
-		}
-
-		stop() {
-			this.output.stop();
-			return this.ac.currentTime - this.lastStartTime;
-		}
-
-		connect(target) {
-			this.target = target;
-		}
-
-		setLoop(loop) {
-			this.loop = loop;
-		}
-	}
-	class AudioPlayer {
-		constructor() {
-			/** 音频播放上下文 */
-			this.ac = new AudioContext();
-			/** 音量节点 */
-			this.gain = this.ac.createGain();
-			this.gain.connect(this.ac.destination);
-			this.audioRoutes = new Map();
-		}
-		/**
-		 * 解码音频数据
-		 * @param data 音频数据
-		 */
-		decodeAudioData(data) {
-			return AudioDecoder.decodeAudioData(data, this);
-		}
-		/**
-		 * 设置音量
-		 * @param volume 音量
-		 */
-		setVolume(volume) {
-			this.gain.gain.value = volume;
-		}
-
-		/**
-		 * 获取音量
-		 */
-		getVolume() {
-			return this.gain.gain.value;
-		}
-
-		/**
-		 * 创建一个音频源
-		 * @param Source 音频源类
-		 */
-		createSource(Source) {
-			return new Source(this.ac);
-		}
-
-		/**
-		 * 创建一个兼容流式音频源,可以与流式加载相结合,主要用于处理 opus ogg 不兼容的情况
-		 */
-		createStreamSource() {
-			return new AudioStreamSource(this.ac);
-		}
-
-		/**
-		 * 创建一个通过 audio 元素播放的音频源
-		 */
-		createElementSource() {
-			return new AudioElementSource(this.ac);
-		}
-
-		/**
-		 * 创建一个通过 AudioBuffer 播放的音频源
-		 */
-		createBufferSource() {
-			return new AudioBufferSource(this.ac);
-		}
-
-		/**
-		 * 获取音频目的地
-		 */
-		getDestination() {
-			return this.gain;
-		}
-
-		/**
-		 * 创建一个音频效果器
-		 * @param Effect 效果器类
-		 */
-		createEffect(Effect) {
-			return new Effect(this.ac);
-		}
-
-		/**
-		 * 创建一个修改音量的效果器
-		 * ```txt
-		 *             |----------|
-		 * Input ----> | GainNode | ----> Output
-		 *             |----------|
-		 * ```
-		 */
-		createVolumeEffect() {
-			return new VolumeEffect(this.ac);
-		}
-
-		/**
-		 * 创建一个立体声效果器
-		 * ```txt
-		 *             |------------|
-		 * Input ----> | PannerNode | ----> Output
-		 *             |------------|
-		 * ```
-		 */
-		createStereoEffect() {
-			return new StereoEffect(this.ac);
-		}
-
-		/**
-		 * 创建一个修改单个声道音量的效果器
-		 * ```txt
-		 *                                  |----------|
-		 *                               -> | GainNode | \
-		 *             |--------------| /   |----------|  -> |------------|
-		 * Input ----> | SplitterNode |        ......        | MergerNode | ----> Output
-		 *             |--------------| \   |----------|  -> |------------|
-		 *                               -> | GainNode | /
-		 *                                  |----------|
-		 * ```
-		 */
-		createChannelVolumeEffect() {
-			return new ChannelVolumeEffect(this.ac);
-		}
-
-		/**
-		 * 创建一个延迟效果器
-		 *             |-----------|
-		 * Input ----> | DelayNode | ----> Output
-		 *             |-----------|
-		 */
-		createDelay() {
-			return new DelayEffect(this.ac);
-		}
-
-		/**
-		 * 创建一个回声效果器
-		 * ```txt
-		 *             |----------|
-		 * Input ----> | GainNode | ----> Output
-		 *        ^    |----------|   |
-		 *        |                   |
-		 *        |   |------------|  ↓
-		 *        |-- | Delay Node | <--
-		 *            |------------|
-		 * ```
-		 */
-		createEchoEffect() {
-			return new EchoEffect(this.ac);
-		}
-
-		/**
-		 * 创建一个音频播放路由
-		 * @param source 音频源
-		 */
-		createRoute(source) {
-			return new AudioRoute(source, this);
-		}
-
-		/**
-		 * 添加一个音频播放路由,可以直接被播放
-		 * @param id 这个音频播放路由的名称
-		 * @param route 音频播放路由对象
-		 */
-		addRoute(id, route) {
-			if (!this.audioRoutes) this.audioRoutes = new Map();
-			if (this.audioRoutes.has(id)) {
-				console.warn(
-					"Audio route with id of '" +
-					id +
-					"' has already existed. New route will override old route."
-				);
-			}
-			this.audioRoutes.set(id, route);
-		}
-
-		/**
-		 * 根据名称获取音频播放路由对象
-		 * @param id 音频播放路由的名称
-		 */
-		getRoute(id) {
-			return this.audioRoutes.get(id);
-		}
-		/**
-		 * 移除一个音频播放路由
-		 * @param id 要移除的播放路由的名称
-		 */
-		removeRoute(id) {
-			this.audioRoutes.delete(id);
-		}
-		/**
-		 * 播放音频
-		 * @param id 音频名称
-		 * @param when 从音频的哪个位置开始播放,单位秒
-		 */
-		play(id, when) {
-			const route = this.getRoute(id);
-			if (!route) {
-				console.warn(
-					"Cannot play audio route '" +
-					id +
-					"', since there is not added route named it."
-				);
-				return;
-			}
-
-			route.play(when);
-		}
-
-		/**
-		 * 暂停音频播放
-		 * @param id 音频名称
-		 * @returns 当音乐真正停止时兑现
-		 */
-		pause(id) {
-			const route = this.getRoute(id);
-			if (!route) {
-				console.warn(
-					"Cannot pause audio route '" +
-					id +
-					"', since there is not added route named it."
-				);
-				return;
-			}
-			return route.pause();
-		}
-
-		/**
-		 * 停止音频播放
-		 * @param id 音频名称
-		 * @returns 当音乐真正停止时兑现
-		 */
-		stop(id) {
-			const route = this.getRoute(id);
-			if (!route) {
-				console.warn(
-					"Cannot stop audio route '" +
-					id +
-					"', since there is not added route named it."
-				);
-				return;
-			}
-			return route.stop();
-		}
-
-		/**
-		 * 继续音频播放
-		 * @param id 音频名称
-		 */
-		resume(id) {
-			const route = this.getRoute(id);
-			if (!route) {
-				console.warn(
-					"Cannot pause audio route '" +
-					id +
-					"', since there is not added route named it."
-				);
-				return;
-			}
-			route.resume();
-		}
-
-		/**
-		 * 设置听者位置,x正方向水平向右,y正方向垂直于地面向上,z正方向垂直屏幕远离用户
-		 * @param x 位置x坐标
-		 * @param y 位置y坐标
-		 * @param z 位置z坐标
-		 */
-		setListenerPosition(x, y, z) {
-			const listener = this.ac.listener;
-			listener.positionX.value = x;
-			listener.positionY.value = y;
-			listener.positionZ.value = z;
-		}
-
-		/**
-		 * 设置听者朝向,x正方向水平向右,y正方向垂直于地面向上,z正方向垂直屏幕远离用户
-		 * @param x 朝向x坐标
-		 * @param y 朝向y坐标
-		 * @param z 朝向z坐标
-		 */
-		setListenerOrientation(x, y, z) {
-			const listener = this.ac.listener;
-			listener.forwardX.value = x;
-			listener.forwardY.value = y;
-			listener.forwardZ.value = z;
-		}
-
-		/**
-		 * 设置听者头顶朝向,x正方向水平向右,y正方向垂直于地面向上,z正方向垂直屏幕远离用户
-		 * @param x 头顶朝向x坐标
-		 * @param y 头顶朝向y坐标
-		 * @param z 头顶朝向z坐标
-		 */
-		setListenerUp(x, y, z) {
-			const listener = this.ac.listener;
-			listener.upX.value = x;
-			listener.upY.value = y;
-			listener.upZ.value = z;
-		}
-	}
-	class AudioRoute {
-		constructor(source, player) {
-			this.output = source.output;
-
-			/** 效果器路由图 */
-			this.effectRoute = [];
-
-			/** 结束时长,当音频暂停或停止时,会经过这么长时间之后才真正终止播放,期间可以做音频淡入淡出等效果 */
-			this.endTime = 0;
-			/** 暂停时播放了多长时间 */
-			this.pauseCurrentTime = 0;
-			/** 当前播放状态 */
-			this.player = player;
-			this.player.status = AudioStatus.Stoped;
-			this.shouldStop = false;
-			/**
-			 * 每次暂停或停止时自增,用于判断当前正在处理的情况。
-			 * 假如暂停后很快播放,然后很快暂停,那么需要根据这个来判断实际是否应该执行暂停后操作
-			 */
-			this.stopIdentifier = 0;
-			/** 暂停时刻 */
-			this.pauseTime = 0;
-			this.source = source;
-			this.source.player = player;
-		}
-		/** 音频时长,单位秒 */
-		get duration() {
-			return this.source.duration;
-		}
-		/** 当前播放了多长时间,单位秒 */
-		get currentTime() {
-			if (this.player.status === AudioStatus.Paused) {
-				return this.pauseCurrentTime;
-			} else {
-				return this.source.currentTime;
-			}
-		}
-		set currentTime(time) {
-			this.source.stop();
-			this.source.play(time);
-		}
-		/**
-		 * 设置结束时间,暂停或停止时,会经过这么长时间才终止音频的播放,这期间可以做一下音频淡出的效果。
-		 * @param time 暂停或停止时,经过多长时间之后才会结束音频的播放
-		 */
-		setEndTime(time) {
-			this.endTime = time;
-		}
-
-		/**
-		 * 当音频播放时执行的函数,可以用于音频淡入效果
-		 * @param fn 音频开始播放时执行的函数
-		 */
-		onStart(fn) {
-			this.audioStartHook = fn;
-		}
-
-		/**
-		 * 当音频暂停或停止时执行的函数,可以用于音频淡出效果
-		 * @param fn 音频在暂停或停止时执行的函数,不填时表示取消这个钩子。
-		 *           包含两个参数,第一个参数是结束时长,第二个参数是当前音频播放路由对象
-		 */
-		onEnd(fn) {
-			this.audioEndHook = fn;
-		}
-
-		/**
-		 * 开始播放这个音频
-		 * @param when 从音频的什么时候开始播放,单位秒
-		 */
-		play(when = 0) {
-			if (this.player.status === AudioStatus.Playing) return;
-			this.link();
-			if (this.effectRoute.length > 0) {
-				const first = this.effectRoute[0];
-				this.source.connect(first);
-				const last = this.effectRoute.at(-1);
-				last.connect({ input: this.player.getDestination() });
-			} else {
-				this.source.connect({ input: this.player.getDestination() });
-			}
-			this.source.play(when);
-			this.player.status = AudioStatus.Playing;
-			this.pauseTime = 0;
-			this.audioStartHook?.(this);
-			this.startAllEffect();
-			if (this.source.player.status !== AudioStatus.Playing) {
-				this.source.player.status = AudioStatus.Playing;
-			}
-		}
-
-		/**
-		 * 暂停音频播放
-		 */
-		async pause() {
-			if (this.player.status !== AudioStatus.Playing) return;
-			this.player.status = AudioStatus.Pausing;
-			this.stopIdentifier++;
-			const identifier = this.stopIdentifier;
-			if (this.audioEndHook) {
-				this.audioEndHook(this.endTime, this);
-				await sleep(this.endTime);
-			}
-			if (
-				this.player.status !== AudioStatus.Pausing ||
-				this.stopIdentifier !== identifier
-			) {
-				return;
-			}
-			this.pauseCurrentTime = this.source.currentTime;
-			const time = this.source.stop();
-			this.pauseTime = time;
-			if (this.shouldStop) {
-				this.player.status = AudioStatus.Stoped;
-				this.endAllEffect();
-
-				this.shouldStop = false;
-			} else {
-				this.player.status = AudioStatus.Paused;
-				this.endAllEffect();
-			}
-			this.endAllEffect();
-		}
-
-		/**
-		 * 继续音频播放
-		 */
-		resume() {
-			if (this.player.status === AudioStatus.Playing) return;
-			if (
-				this.player.status === AudioStatus.Pausing ||
-				this.player.status === AudioStatus.Stoping
-			) {
-				this.audioStartHook?.(this);
-
-				return;
-			}
-			if (this.player.status === AudioStatus.Paused) {
-				this.play(this.pauseTime);
-			} else {
-				this.play(0);
-			}
-			this.player.status = AudioStatus.Playing;
-			this.pauseTime = 0;
-			this.audioStartHook?.(this);
-			this.startAllEffect();
-		}
-
-		/**
-		 * 停止音频播放
-		 */
-		async stop() {
-			if (this.status !== AudioStatus.Playing) {
-				if (this.status === AudioStatus.Pausing) {
-					this.shouldStop = true;
-				}
-				return;
-			}
-			this.status = AudioStatus.Stoping;
-			this.stopIdentifier++;
-			const identifier = this.stopIdentifier;
-			if (this.audioEndHook) {
-				this.audioEndHook(this.endTime, this);
-				await sleep(this.endTime);
-			}
-			if (
-				this.status !== AudioStatus.Stoping ||
-				this.stopIdentifier !== identifier
-			) {
-				return;
-			}
-			this.source.stop();
-			this.status = AudioStatus.Stoped;
-			this.pauseTime = 0;
-			this.endAllEffect();
-		}
-
-		/**
-		 * 添加效果器
-		 * @param effect 要添加的效果,可以是数组,表示一次添加多个
-		 * @param index 从哪个位置开始添加,如果大于数组长度,那么加到末尾,如果小于0,那么将会从后面往前数。默认添加到末尾
-		 */
-		addEffect(effect, index) {
-			if (isNil(index)) {
-				if (effect instanceof Array) {
-					this.effectRoute.push(...effect);
-				} else {
-					this.effectRoute.push(effect);
-				}
-			} else {
-				if (effect instanceof Array) {
-					this.effectRoute.splice(index, 0, ...effect);
-				} else {
-					this.effectRoute.splice(index, 0, effect);
-				}
-			}
-			this.setOutput();
-			if (this.source.playing) this.link();
-		}
-
-		/**
-		 * 移除一个效果器
-		 * @param effect 要移除的效果
-		 */
-		removeEffect(effect) {
-			const index = this.effectRoute.indexOf(effect);
-			if (index === -1) return;
-			this.effectRoute.splice(index, 1);
-			effect.disconnect();
-			this.setOutput();
-			if (this.source.playing) this.link();
-		}
-
-		setOutput() {
-			const effect = this.effectRoute.at(-1);
-			if (!effect) this.output = this.source.output;
-			else this.output = effect.output;
-		}
-
-		/**
-		 * 连接音频路由图
-		 */
-		link() {
-			this.effectRoute.forEach((v) => v.disconnect());
-			this.effectRoute.forEach((v, i) => {
-				const next = this.effectRoute[i + 1];
-				if (next) {
-					v.connect(next);
-				}
-			});
-		}
-
-		startAllEffect() {
-			this.effectRoute.forEach((v) => v.start());
-		}
-
-		endAllEffect() {
-			this.effectRoute.forEach((v) => v.end());
-		}
-	}
-
-	const audioPlayer = new AudioPlayer();
-
-	class BgmController {
-		constructor(player) {
-			this.mainGain = player.createVolumeEffect();
-			this.player = player;
-			/** bgm音频名称的前缀 */
-			this.prefix = "bgms.";
-			/** 每个 bgm 的音量控制器 */
-			this.gain = new Map();
-
-			/** 正在播放的 bgm */
-			this.playingBgm = "";
-			/** 是否正在播放 */
-			this.playing = false;
-
-			/** 是否已经启用 */
-			this.enabled = true;
-			/** 是否屏蔽所有的音乐切换 */
-			this.blocking = false;
-			/** 渐变时长 */
-			this.transitionTime = 2000;
-		}
-
-		/**
-		 * 设置音频渐变时长
-		 * @param time 渐变时长
-		 */
-		setTransitionTime(time) {
-			this.transitionTime = time;
-			for (const [, value] of this.gain) {
-				value.transition.time(time);
-			}
-		}
-
-		/**
-		 * 屏蔽音乐切换
-		 */
-		blockChange() {
-			this.blocking = true;
-		}
-
-		/**
-		 * 取消屏蔽音乐切换
-		 */
-		unblockChange() {
-			this.blocking = false;
-		}
-
-		/**
-		 * 设置总音量大小
-		 * @param volume 音量大小
-		 */
-		setVolume(volume) {
-			this.mainGain.setVolume(volume);
-			this._volume = volume;
-		}
-		/**
-		 * 获取总音量大小
-		 */
-		getVolume() {
-			return this.mainGain.getVolume();
-		}
-		/**
-		 * 设置是否启用
-		 * @param enabled 是否启用
-		 */
-		setEnabled(enabled) {
-			if (enabled) this.resume();
-			else this.stop();
-			this.enabled = enabled;
-		}
-
-		/**
-		 * 设置 bgm 音频名称的前缀
-		 */
-		setPrefix(prefix) {
-			this.prefix = prefix;
-		}
-
-		getId(name) {
-			return `${this.prefix}${name}`;
-		}
-
-		/**
-		 * 根据 bgm 名称获取其 AudioRoute 实例
-		 * @param id 音频名称
-		 */
-		get(id) {
-			return this.player.getRoute(this.getId(id));
-		}
-
-		/**
-		 * 添加一个 bgm
-		 * @param id 要添加的 bgm 的名称
-		 * @param url 指定 bgm 的加载地址
-		 */
-		addBgm(id, url = `project/bgms/${id}`) {
-			const type = guessTypeByExt(id);
-			if (!type) {
-				console.warn(
-					"Unknown audio extension name: '" +
-					id.split(".").slice(0, -1).join(".") +
-					"'"
-				);
-				return;
-			}
-			const gain = this.player.createVolumeEffect();
-			if (isAudioSupport(type)) {
-				const source = audioPlayer.createElementSource();
-				source.setSource(url);
-				source.setLoop(true);
-				const route = new AudioRoute(source, audioPlayer);
-				route.addEffect([gain, this.mainGain]);
-				audioPlayer.addRoute(this.getId(id), route);
-				this.setTransition(id, route, gain);
-			} else {
-				const source = audioPlayer.createStreamSource();
-				const stream = new StreamLoader(url);
-				stream.pipe(source);
-				source.setLoop(true);
-				const route = new AudioRoute(source, audioPlayer);
-				route.addEffect([gain, this.mainGain]);
-				audioPlayer.addRoute(this.getId(id), route);
-				this.setTransition(id, route, gain);
-			}
-		}
-
-		/**
-		 * 移除一个 bgm
-		 * @param id 要移除的 bgm 的名称
-		 */
-		removeBgm(id) {
-			this.player.removeRoute(this.getId(id));
-			const gain = this.gain.get(id);
-			gain?.transition.ticker.destroy();
-			this.gain.delete(id);
-		}
-
-		setTransition(id, route, gain) {
-			const transition = new Transition();
-			transition
-				.time(this.transitionTime)
-				.mode(linear())
-				.transition("volume", 0);
-
-			const tick = () => {
-				gain.setVolume(transition.value.volume);
-			};
-
-			/**
-			 * @param expect 在结束时应该是正在播放还是停止
-			 */
-			const setTick = async (expect) => {
-				transition.ticker.remove(tick);
-				transition.ticker.add(tick);
-				const identifier = route.stopIdentifier;
-				await sleep(this.transitionTime + 500);
-				if (route.status === expect && identifier === route.stopIdentifier) {
-					transition.ticker.remove(tick);
-					if (route.status === AudioStatus.Playing) {
-						gain.setVolume(1);
-					} else {
-						gain.setVolume(0);
-					}
-				}
-			};
-
-			route.onStart(async () => {
-				transition.transition("volume", 1);
-				setTick(AudioStatus.Playing);
-			});
-			route.onEnd(() => {
-				transition.transition("volume", 0);
-				setTick(AudioStatus.Paused);
-			});
-			route.setEndTime(this.transitionTime);
-
-			this.gain.set(id, { effect: gain, transition });
-		}
-
-		/**
-		 * 播放一个 bgm
-		 * @param id 要播放的 bgm 名称
-		 */
-		play(id, when) {
-			if (this.blocking) return;
-			if (id !== this.playingBgm && this.playingBgm) {
-				this.player.pause(this.getId(this.playingBgm));
-			}
-			this.playingBgm = id;
-			if (!this.enabled) return;
-			this.player.play(this.getId(id), when);
-			this.playing = true;
-			if (this.player.status !== AudioStatus.Playing) {
-				this.player.status = AudioStatus.Playing;
-			}
-		}
-
-		/**
-		 * 继续当前的 bgm
-		 */
-		resume() {
-			if (this.blocking || !this.enabled || this.playing) return;
-			if (this.playingBgm) {
-				this.player.resume(this.getId(this.playingBgm));
-			}
-			this.playing = true;
-		}
-
-		/**
-		 * 暂停当前的 bgm
-		 */
-		pause() {
-			if (this.blocking || !this.enabled) return;
-			if (this.playingBgm) {
-				this.player.pause(this.getId(this.playingBgm));
-			}
-			this.playing = false;
-		}
-
-		/**
-		 * 停止当前的 bgm
-		 */
-		stop() {
-			if (this.blocking || !this.enabled) return;
-			if (this.playingBgm) {
-				this.player.stop(this.getId(this.playingBgm));
-			}
-			this.playing = false;
-		}
-	}
-	const bgmController = new BgmController(audioPlayer);
-
-	class SoundPlayer {
-		constructor(player) {
-			/** 每个音效的唯一标识符 */
-			this.num = 0;
-			this.enabled = true;
-			this.gain = player.createVolumeEffect();
-			/** 每个音效的数据 */
-			this.buffer = new Map();
-			/** 所有正在播放的音乐 */
-			this.playing = new Set();
-			this.player = player;
-		}
-		/**
-		 * 设置是否启用音效
-		 * @param enabled 是否启用音效
-		 */
-		setEnabled(enabled) {
-			if (!enabled) this.stopAllSounds();
-			this.enabled = enabled;
-		}
-
-		/**
-		 * 设置音量大小
-		 * @param volume 音量大小
-		 */
-		setVolume(volume) {
-			this.gain.setVolume(volume);
-		}
-		/**
-		 * 获取音量大小
-		 */
-		getVolume() {
-			return this.gain.getVolume();
-		}
-		/**
-		 * 添加一个音效
-		 * @param id 音效名称
-		 * @param data 音效的Uint8Array数据
-		 */
-		async add(id, data) {
-			const buffer = await this.player.decodeAudioData(data);
-			if (!buffer) {
-				console.warn(
-					"Cannot decode sound '" +
-					id +
-					"', since audio file may not supported by 2.b."
-				);
-				return;
-			}
-			this.buffer.set(id, buffer);
-		}
-
-		/**
-		 * 播放一个音效
-		 * @param id 音效名称
-		 * @param position 音频位置,[0, 0, 0]表示正中心,x轴指向水平向右,y轴指向水平向上,z轴指向竖直向上
-		 * @param orientation 音频朝向,[0, 1, 0]表示朝向前方
-		 */
-		play(id, position = [0, 0, 0], orientation = [1, 0, 0]) {
-			if (!this.enabled) return -1;
-			const buffer = this.buffer.get(id);
-			if (!buffer) {
-				console.warn(
-					"Cannot play sound '" +
-					id +
-					"', since there is no added data named it."
-				);
-				return -1;
-			}
-			const soundNum = this.num++;
-
-			const source = this.player.createBufferSource();
-			source.setBuffer(buffer);
-			const route = this.player.createRoute(source);
-			const stereo = this.player.createStereoEffect();
-			stereo.setPosition(position[0], position[1], position[2]);
-			stereo.setOrientation(orientation[0], orientation[1], orientation[2]);
-			route.addEffect([stereo, this.gain]);
-			this.player.addRoute(`sounds.${soundNum}`, route);
-			route.play();
-			source.output.addEventListener("ended", () => {
-				this.playing.delete(soundNum);
-			});
-			this.playing.add(soundNum);
-			return soundNum;
-		}
-
-		/**
-		 * 停止一个音效
-		 * @param num 音效的唯一 id
-		 */
-		stop(num) {
-			const id = `sounds.${num}`;
-			const route = this.player.getRoute(id);
-			if (route) {
-				route.stop();
-				this.player.removeRoute(id);
-				this.playing.delete(num);
-			}
-		}
-
-		/**
-		 * 停止播放所有音效
-		 */
-		stopAllSounds() {
-			this.playing.forEach((v) => {
-				const id = `sounds.${v}`;
-				const route = this.player.getRoute(id);
-				if (route) {
-					route.stop();
-					this.player.removeRoute(id);
-				}
-			});
-			this.playing.clear();
-		}
-	}
-	const soundPlayer = new SoundPlayer(audioPlayer);
-
-	function loadAllBgm() {
-		const data = data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d;
-		for (const bgm of data.main.bgms) {
-			bgmController.addBgm(bgm);
-		}
-	}
-	loadAllBgm();
-	AudioDecoder.registerDecoder(AudioType.Ogg, VorbisDecoder);
-	AudioDecoder.registerDecoder(AudioType.Opus, OpusDecoder);
-
-	core.plugin.audioSystem = {
-		AudioType,
-		AudioDecoder,
-		AudioStatus,
-		checkAudioType,
-		isAudioSupport,
-		audioPlayer,
-		soundPlayer,
-		bgmController,
-		guessTypeByExt,
-		BgmController,
-		SoundPlayer,
-		EchoEffect,
-		DelayEffect,
-		ChannelVolumeEffect,
-		VolumeEffect,
-		StereoEffect,
-		AudioEffect,
-		AudioPlayer,
-		AudioRoute,
-		AudioStreamSource,
-		AudioElementSource,
-		AudioBufferSource,
-		loadAllBgm,
-		StreamLoader,
-	};
-	//bgm相关复写
-	control.prototype.playBgm = (bgm, when) => {
-		bgmController.play(bgm, when);
-		core.setMusicBtn();
-	};
-	control.prototype.pauseBgm = () => {
-		bgmController.pause();
-		core.setMusicBtn();
-	};
-
-	control.prototype.resumeBgm = function () {
-		bgmController.resume();
-		core.setMusicBtn();
-	};
-	control.prototype.checkBgm = function () {
-		if (bgmController.playing) return;
-		if (core.musicStatus.bgmStatus) {
-			if (bgmController.playingBgm) {
-				bgmController.play(bgmController.playingBgm);
-			} else {
-				play(main.startBgm, 0);
-			}
-		} else {
-			pause();
-		}
-	};
-	control.prototype.triggerBgm = function () {
-		core.musicStatus.bgmStatus = !core.musicStatus.bgmStatus;
-		if (bgmController.playing) bgmController.pause();
-		else bgmController.resume();
-		core.setMusicBtn();
-		core.setLocalStorage("bgmStatus", core.musicStatus.bgmStatus);
-	};
-	//sound相关复写
-	control.prototype.playSound = function (
-		sound,
-		_pitch,
-		callback,
-		position,
-		orientation
-	) {
-		if (main.mode != "play" || !core.musicStatus.soundStatus) return;
-		const name = core.getMappedName(sound);
-		const num = soundPlayer.play(name, position, orientation);
-		const route = audioPlayer.getRoute(`sounds.${num}`);
-		if (!route) {
-			callback?.();
-			return -1;
-		} else {
-			sleep(route.duration).then(() => callback?.());
-			return num;
-		}
-	};
-	control.prototype.stopSound = function (id) {
-		if (isNil(id)) {
-			soundPlayer.stopAllSounds();
-		} else {
-			soundPlayer.stop(id);
-		}
-	};
-	control.prototype.getPlayingSounds = function () {
-		return [...soundPlayer.playing];
-	};
-	//sound加载复写
-	loader.prototype._loadOneSound_decodeData = function (name, data) {
-		if (data instanceof Blob) {
-			var blobReader = new zip.BlobReader(data);
-			blobReader.init(function () {
-				blobReader.readUint8Array(0, blobReader.size, function (uint8) {
-					//core.loader._loadOneSound_decodeData(name, uint8.buffer);
-					soundPlayer.add(name, uint8);
-				});
-			});
-			return;
-		}
-		if (data instanceof ArrayBuffer) {
-			const uint8 = new Uint8Array(data);
-			soundPlayer.add(name, uint8);
-		}
-	};
-	//音量控制复写
-	soundPlayer.setVolume(
-		core.musicStatus.userVolume * core.musicStatus.designVolume
-	);
-	bgmController.setVolume(
-		core.musicStatus.userVolume * core.musicStatus.designVolume
-	);
-	actions.prototype._clickSwitchs_sounds_userVolume = function (delta) {
-		var value = Math.round(Math.sqrt(100 * core.musicStatus.userVolume));
-		if (value == 0 && delta < 0) return;
-		core.musicStatus.userVolume = core.clamp(
-			Math.pow(value + delta, 2) / 100,
-			0,
-			1
-		);
-		//audioContext 音效 不受designVolume 影响
-		if (core.musicStatus.gainNode != null)
-			core.musicStatus.gainNode.gain.value = core.musicStatus.userVolume;
-		soundPlayer.setVolume(
-			core.musicStatus.userVolume * core.musicStatus.designVolume
-		);
-		bgmController.setVolume(
-			core.musicStatus.userVolume * core.musicStatus.designVolume
-		);
-		core.setLocalStorage("userVolume", core.musicStatus.userVolume);
-		core.playSound("确定");
-		core.ui._drawSwitchs_sounds();
-	};
-},
-    "怪物碎裂特效": function () {
-    // 在此增加新插件
-    // -------------------- 安装说明 -------------------- //
-    // 先安装两个在插件简介中说明的前置插件
-    // 然后再将本插件复制到插件编写中即可
-    // 插件自带一个打怪后显示碎裂特效的功能
-    // -------------------- 使用说明 -------------------- //
-    /*
-
-	本插件的核心是一个名为 applyFragWith 的函数,打怪后的碎裂特效也是由它执行的。
-	我们来说明一下这个函数的使用方式。
-	1. 引入
-	  你可以使用 const { applyFragWith } = core.plugin.frag; 在任何地方来引入这个函数。
-
-	2. 函数的参数
-	  该函数有三个参数,分别是canvas, length, time。
-	  其中,第一个参数意思是,在执行碎裂时,其内容由该画布决定,这个特效并不会修改传入的画布,因此原画布的内容会依然存在
-	  第二个参数指的是每个碎片的边长,虽然原则上每个碎片都是正方形,但边缘一周的碎片可能不是正方形,但中间的碎片一定是
-	  第三个参数指的是这个特效要执行多长时间
-	  第四个参数是一个对象,包含四种配置量,均为可选,分别是maxMoveLength,moveFlush,maxRotate,fragTiming
-	  当这些不存在时会默认取同名的常量作为默认值。这些值有什么用可以看下面的常量注释。
-
-	3. 函数的返回值
-	  这个函数会返回一个特效控制器对象,这个控制器共有三个属性。
-	  animation: 指的是当前特效的高级动画对象
-	  onEnd: 一个Promise,当这个特效执行完毕后会被 fulfilled
-	  canvas: 特效所显示的画布。这个画布不会自动部署到样板中,需要你手动使用appendChild来部署,具体可参考打怪后碎裂的样例
-
-	4. 修改一些常量
-	  在下面有四个有注释的常量MAX_MOVE_LENGTH ~ FRAG_TIMING,你可以根据你自己的需要来更改。
-
-	*/
-
-    if (main.replayChecking) return (core.plugin.frag = {});
-
-    const { Animation, linear, sleep } = core.plugin.animate;
-    const { has } = core.plugin.utils;
-
-    /** 最大移动距离,最终位置距离中心的距离变成原来的几倍 */
-    const MAX_MOVE_LENGTH = 1.15;
-    /** 移动距离波动,在最大移动距离的基础上加上多少倍距离的波动距离 */
-    const MOVE_FLUSH = 0.7;
-    /** 最大旋转角,单位是弧度,每个碎片都会有自己的旋转程度,是随机的 */
-    const MAX_ROTATE = 0.5;
-    /** 碎裂动画的速率曲线函数 */
-    const FRAG_TIMING = linear();
-
+    // 将__enable置为false将关闭插件
+    let __enable = true;
+    if (!__enable || main.mode === "editor") return;
+    const { OggOpusDecoderWebWorker } = window["ogg-opus-decoder"];
+    const { OggVorbisDecoderWebWorker } = window["ogg-vorbis-decoder"];
+    const { CodecParser } = window.CodecParser;
+    const { Transition, linear } = core.plugin.animate;
+
+    const audio = new Audio();
+    const AudioStatus = {
+      Playing: 0,
+      Pausing: 1,
+      Paused: 2,
+      Stoping: 3,
+      Stoped: 4,
+    };
+    const supportMap = new Map();
+    const AudioType = {
+      Mp3: "audio/mpeg",
+      Wav: 'audio/wav; codecs="1"',
+      Flac: "audio/flac",
+      Opus: 'audio/ogg; codecs="opus"',
+      Ogg: 'audio/ogg; codecs="vorbis"',
+      Aac: "audio/aac",
+    };
     /**
-     * @param {HTMLCanvasElement} canvas 要执行特效的画布
-     * @param {number} length 切分成的碎片的边长,碎片为正方形
-     * @param {number} time 特效持续时长
-     * @returns 返回一个碎裂特效控制器,是一个对象,详见开头的使用注释
+     * 检查一种音频类型是否能被播放
+     * @param type 音频类型 AudioType
      */
-    function applyFragWith(canvas, length = 4, time = 2000, config = {}) {
-      // 先切分图片
-      const imgs = splitCanvas(canvas, length);
-      const cx = canvas.width / 2;
-      const cy = canvas.height / 2;
-
-      let maxX = 0;
-      let maxY = 0;
-      const toMove = imgs.map((v) => {
-        const centerX = v.x + v.canvas.width / 2;
-        const centerY = v.y + v.canvas.height / 2;
-        const onX = centerX === cx;
-        const onY = centerY === cy;
-        const mml = config.maxMoveLength ?? MAX_MOVE_LENGTH;
-        const mf = config.moveFlush ?? MOVE_FLUSH;
-        const rate = mml - 1 + Math.random() ** 3 * mf;
-        let endX = onY ? 0 : (centerX - cx) * rate;
-        let endY = onX ? 0 : (centerY - cy) * rate;
-        const mx = Math.abs(endX + centerX) + Math.abs(v.canvas.width);
-        const my = Math.abs(endY + centerY) + Math.abs(v.canvas.height);
-        if (mx > maxX) maxX = mx;
-        if (my > maxY) maxY = my;
-        const r = config.maxRotate ?? MAX_ROTATE;
-        const endRad = Math.random() * r * 2 - r;
-
-        return {
-          deltaX: endX,
-          deltaY: endY,
-          endRad,
-          x: centerX,
-          y: centerY,
-          canvas: v.canvas,
-        };
-      });
-
-      // 再执行动画
-      const frag = document.createElement("canvas");
-      const ctx = frag.getContext("2d");
-      const ani = new Animation();
-      ani.register("rate", 0);
-      const ft = config.fragTiming ?? FRAG_TIMING;
-      ani.absolute().time(time).mode(ft).apply("rate", 1);
-      frag.width = maxX * 2;
-      frag.height = maxY * 2;
-      ctx.save();
-      const dw = maxX - canvas.width / 2;
-      const dh = maxY - canvas.height / 2;
-
-      const fragFn = () => {
-        const rate = ani.value.rate;
-        const opacity = 1 - rate;
-        ctx.globalAlpha = opacity;
-        ctx.clearRect(0, 0, frag.width, frag.height);
-        toMove.forEach((v) => {
-          ctx.save();
-          const nx = v.deltaX * rate;
-          const ny = v.deltaY * rate;
-          const rotate = v.endRad * rate;
-
-          ctx.translate(nx + v.x + dw, ny + v.y + dh);
-          ctx.rotate(rotate);
-          ctx.drawImage(
-            v.canvas,
-            nx - v.canvas.width / 2,
-            ny - v.canvas.height / 2
-          );
-          ctx.restore();
-        });
-      };
-      const onEnd = () => {};
-      ani.ticker.add(fragFn);
-
-      return makeFragManager(frag, ani, time, onEnd);
-    }
-
-    function makeFragManager(canvas, ani, time, onEnd) {
-      const promise = sleep(time + 50);
-
-      return {
-        animation: ani,
-
-        onEnd: promise.then(() => {
-          ani.ticker.destroy();
-          onEnd();
-        }),
-        canvas,
-      };
-    }
-
-    function withImage(image, sx, sy, sw, sh) {
-      const canvas = document.createElement("canvas");
-      const ctx = canvas.getContext("2d");
-      canvas.width = sw;
-      canvas.height = sh;
-      ctx.drawImage(image, sx, sy, sw, sh, 0, 0, sw, sh);
-      return { canvas, x: sx, y: sy };
-    }
-
-    /**
-     * 切分画布
-     * @param canvas 要被切分的画布
-     * @param l 切分小块的边长
-     */
-    function splitCanvas(canvas, l) {
-      if (canvas.width / l < 2 || canvas.height / l < 2) {
-        console.warn("切分画布要求切分边长大于等于画布长宽的一半!");
-        return [];
+    function isAudioSupport(type) {
+      if (supportMap.has(type)) return supportMap.get(type);
+      else {
+        const support = audio.canPlayType(type);
+        const canPlay = support === "maybe" || support === "probably";
+        supportMap.set(type, canPlay);
+        return canPlay;
       }
-      const w = canvas.width;
-      const h = canvas.height;
-      const numX = Math.floor(w / l);
-      const numY = Math.floor(h / l);
-      const rw = (w - numX * l) / 2;
-      const rh = (h - numY * l) / 2;
+    }
 
-      const res = [];
+    const typeMap = new Map([
+      ["ogg", AudioType.Ogg],
+      ["mp3", AudioType.Mp3],
+      ["wav", AudioType.Wav],
+      ["flac", AudioType.Flac],
+      ["opus", AudioType.Opus],
+      ["aac", AudioType.Aac],
+    ]);
 
-      if (rw > 0) {
-        if (rh > 0) {
-          res.push(
-            withImage(canvas, 0, 0, rw, rh),
-            withImage(canvas, 0, canvas.height - rh, rw, rh),
-            withImage(canvas, canvas.width - rw, 0, rw, rh),
-            withImage(canvas, canvas.width - rw, canvas.height - rh, rw, rh)
-          );
-        }
-        for (const x of [0, canvas.width - rw]) {
-          for (let ny = 0; ny < numY; ny++) {
-            res.push(withImage(canvas, x, rh + l * ny, rw, l));
+    /**
+     * 根据文件名拓展猜测其类型
+     * @param file 文件名 string
+     */
+    function guessTypeByExt(file) {
+      const ext = /\.[a-zA-Z\d]+$/.exec(file);
+      if (!ext?.[0]) return "";
+      const type = ext[0].slice(1);
+      return typeMap.get(type.toLocaleLowerCase()) ?? "";
+    }
+
+    isAudioSupport(AudioType.Ogg);
+    isAudioSupport(AudioType.Mp3);
+    isAudioSupport(AudioType.Wav);
+    isAudioSupport(AudioType.Flac);
+    isAudioSupport(AudioType.Opus);
+    isAudioSupport(AudioType.Aac);
+
+    function isNil(value) {
+      return value === void 0 || value === null;
+    }
+
+    function sleep(time) {
+      return new Promise((res) => setTimeout(res, time));
+    }
+    class AudioEffect {
+      constructor(ac) {}
+      /**
+       * 连接至其他效果器
+       * @param target 目标输入 IAudioInput
+       * @param output 当前效果器输出通道 Number
+       * @param input 目标效果器的输入通道 Number
+       */
+      connect(target, output, input) {
+        this.output.connect(target.input, output, input);
+      }
+
+      /**
+       * 与其他效果器取消连接
+       * @param target 目标输入 IAudioInput
+       * @param output 当前效果器输出通道 Number
+       * @param input 目标效果器的输入通道 Number
+       */
+      disconnect(target, output, input) {
+        if (!target) {
+          if (!isNil(output)) {
+            this.output.disconnect(output);
+          } else {
+            this.output.disconnect();
+          }
+        } else {
+          if (!isNil(output)) {
+            if (!isNil(input)) {
+              this.output.disconnect(target.input, output, input);
+            } else {
+              this.output.disconnect(target.input, output);
+            }
+          } else {
+            this.output.disconnect(target.input);
           }
         }
       }
-      if (rh > 0) {
-        for (const y of [0, canvas.height - rh]) {
-          for (let nx = 0; nx < numX; nx++) {
-            res.push(withImage(canvas, rw + l * nx, y, l, rh));
+    }
+
+    class StereoEffect extends AudioEffect {
+      constructor(ac) {
+        super(ac);
+        const panner = ac.createPanner();
+        this.input = panner;
+        this.output = panner;
+      }
+
+      /**
+       * 设置音频朝向,x正方形水平向右,y正方形垂直于地面向上,z正方向垂直屏幕远离用户
+       * @param x 朝向x坐标 Number
+       * @param y 朝向y坐标 Number
+       * @param z 朝向z坐标 Number
+       */
+      setOrientation(x, y, z) {
+        this.output.orientationX.value = x;
+        this.output.orientationY.value = y;
+        this.output.orientationZ.value = z;
+      }
+      /**
+       * 设置音频位置,x正方形水平向右,y正方形垂直于地面向上,z正方向垂直屏幕远离用户
+       * @param x 位置x坐标 Number
+       * @param y 位置y坐标 Number
+       * @param z 位置z坐标 Number
+       */
+      setPosition(x, y, z) {
+        this.output.positionX.value = x;
+        this.output.positionY.value = y;
+        this.output.positionZ.value = z;
+      }
+      end() {}
+
+      start() {}
+    }
+    class VolumeEffect extends AudioEffect {
+      constructor(ac) {
+        super(ac);
+        const gain = ac.createGain();
+        this.input = gain;
+        this.output = gain;
+      }
+
+      /**
+       * 设置音量大小
+       * @param volume 音量大小 Number
+       */
+      setVolume(volume) {
+        this.output.gain.value = volume;
+      }
+
+      /**
+       * 获取音量大小 Number
+       */
+      getVolume() {
+        return this.output.gain.value;
+      }
+
+      end() {}
+
+      start() {}
+    }
+    class ChannelVolumeEffect extends AudioEffect {
+      /** 所有的音量控制节点 */
+
+      constructor(ac) {
+        super(ac);
+        /** 所有的音量控制节点 */
+        this.gain = [];
+        const splitter = ac.createChannelSplitter();
+        const merger = ac.createChannelMerger();
+        this.output = merger;
+        this.input = splitter;
+        for (let i = 0; i < 6; i++) {
+          const gain = ac.createGain();
+          splitter.connect(gain, i);
+          gain.connect(merger, 0, i);
+          this.gain.push(gain);
+        }
+      }
+
+      /**
+       * 设置某个声道的音量大小
+       * @param channel 要设置的声道,可填0-5 Number
+       * @param volume 这个声道的音量大小 Number
+       */
+      setVolume(channel, volume) {
+        if (!this.gain[channel]) return;
+        this.gain[channel].gain.value = volume;
+      }
+
+      /**
+       * 获取某个声道的音量大小,可填0-5
+       * @param channel 要获取的声道 Number
+       */
+      getVolume(channel) {
+        if (!this.gain[channel]) return 0;
+        return this.gain[channel].gain.value;
+      }
+
+      end() {}
+
+      start() {}
+    }
+    class DelayEffect extends AudioEffect {
+      constructor(ac) {
+        super(ac);
+
+        const delay = ac.createDelay();
+        this.input = delay;
+        this.output = delay;
+      }
+
+      /**
+       * 设置延迟时长
+       * @param delay 延迟时长,单位秒 Number
+       */
+      setDelay(delay) {
+        this.output.delayTime.value = delay;
+      }
+
+      /**
+       * 获取延迟时长
+       */
+      getDelay() {
+        return this.output.delayTime.value;
+      }
+
+      end() {}
+
+      start() {}
+    }
+    class EchoEffect extends AudioEffect {
+      constructor(ac) {
+        super(ac);
+        /** 当前增益 */
+        this.gain = 0.5;
+        /** 是否正在播放 */
+        this.playing = false;
+        const delay = ac.createDelay();
+        const gain = ac.createGain();
+        gain.gain.value = 0.5;
+        delay.delayTime.value = 0.05;
+        delay.connect(gain);
+        gain.connect(delay);
+        /** 延迟节点 */
+        this.delay = delay;
+        /** 反馈增益节点 */
+        this.gainNode = gain;
+
+        this.input = gain;
+        this.output = gain;
+      }
+
+      /**
+       * 设置回声反馈增益大小
+       * @param gain 增益大小,范围 0-1,大于等于1的视为0.5,小于0的视为0 Number
+       */
+      setFeedbackGain(gain) {
+        const resolved = gain >= 1 ? 0.5 : gain < 0 ? 0 : gain;
+        this.gain = resolved;
+        if (this.playing) this.gainNode.gain.value = resolved;
+      }
+
+      /**
+       * 设置回声间隔时长
+       * @param delay 回声时长,范围 0.01-Infinity,小于0.01的视为0.01 Number
+       */
+      setEchoDelay(delay) {
+        const resolved = delay < 0.01 ? 0.01 : delay;
+        this.delay.delayTime.value = resolved;
+      }
+
+      /**
+       * 获取反馈节点增益
+       */
+      getFeedbackGain() {
+        return this.gain;
+      }
+
+      /**
+       * 获取回声间隔时长
+       */
+      getEchoDelay() {
+        return this.delay.delayTime.value;
+      }
+
+      end() {
+        this.playing = false;
+        const echoTime = Math.ceil(Math.log(0.001) / Math.log(this.gain)) + 10;
+        sleep(this.delay.delayTime.value * echoTime).then(() => {
+          if (!this.playing) this.gainNode.gain.value = 0;
+        });
+      }
+
+      start() {
+        this.playing = true;
+        this.gainNode.gain.value = this.gain;
+      }
+    }
+
+    class StreamLoader {
+      constructor(url) {
+        /** 传输目标  Set<IStreamReader>*/
+        this.target = new Set();
+        this.loading = false;
+      }
+
+      /**
+       * 将加载流传递给字节流读取对象
+       * @param reader 字节流读取对象 IStreamReader
+       */
+      pipe(reader) {
+        if (this.loading) {
+          console.warn(
+            "Cannot pipe new StreamReader object when stream is loading."
+          );
+          return;
+        }
+        this.target.add(reader);
+        reader.piped(this);
+        return this;
+      }
+
+      async start() {
+        if (this.loading) return;
+        this.loading = true;
+        const response = await window.fetch(this.url);
+        const stream = response.body;
+        if (!stream) {
+          console.error("Cannot get reader when fetching '" + this.url + "'.");
+          return;
+        }
+        // 获取读取器
+        this.stream = stream;
+        const reader = response.body?.getReader();
+        const targets = [...this.target];
+
+        await Promise.all(targets.map((v) => v.start(stream, this, response)));
+        if (reader && reader.read) {
+          // 开始流传输
+          while (true) {
+            const { value, done } = await reader.read();
+            await Promise.all(
+              targets.map((v) => v.pump(value, done, response))
+            );
+            if (done) break;
+          }
+        } else {
+          // 如果不支持流传输
+          const buffer = await response.arrayBuffer();
+          const data = new Uint8Array(buffer);
+          await Promise.all(targets.map((v) => v.pump(data, true, response)));
+        }
+
+        this.loading = false;
+        targets.forEach((v) => v.end(true));
+
+        //
+      }
+
+      cancel(reason) {
+        if (!this.stream) return;
+        this.stream.cancel(reason);
+        this.loading = false;
+        this.target.forEach((v) => v.end(false, reason));
+      }
+    }
+    const fileSignatures = [
+      [AudioType.Mp3, [0x49, 0x44, 0x33]],
+      [AudioType.Ogg, [0x4f, 0x67, 0x67, 0x53]],
+      [AudioType.Wav, [0x52, 0x49, 0x46, 0x46]],
+      [AudioType.Flac, [0x66, 0x4c, 0x61, 0x43]],
+      [AudioType.Aac, [0xff, 0xf1]],
+      [AudioType.Aac, [0xff, 0xf9]],
+    ];
+    const oggHeaders = [
+      [AudioType.Opus, [0x4f, 0x70, 0x75, 0x73, 0x48, 0x65, 0x61, 0x64]],
+    ];
+
+    function checkAudioType(data) {
+      let audioType = "";
+      // 检查头文件获取音频类型,仅检查前256个字节
+      const toCheck = data.slice(0, 256);
+      for (const [type, value] of fileSignatures) {
+        if (value.every((v, i) => toCheck[i] === v)) {
+          audioType = type;
+          break;
+        }
+      }
+      if (audioType === AudioType.Ogg) {
+        // 如果是ogg的话,进一步判断是不是opus
+        for (const [key, value] of oggHeaders) {
+          const has = toCheck.some((_, i) => {
+            return value.every((v, ii) => toCheck[i + ii] === v);
+          });
+          if (has) {
+            audioType = key;
+            break;
           }
         }
       }
-      for (let nx = 0; nx < numX; nx++) {
-        for (let ny = 0; ny < numY; ny++) {
-          res.push(withImage(canvas, rw + l * nx, rh + l * ny, l, l));
+
+      return audioType;
+    }
+    class AudioDecoder {
+      /**
+       * 注册一个解码器
+       * @param type 要注册的解码器允许解码的类型
+       * @param decoder 解码器对象
+       */
+      static registerDecoder(type, decoder) {
+        if (!this.decoderMap) this.decoderMap = new Map();
+        if (this.decoderMap.has(type)) {
+          console.warn(
+            "Audio stream decoder for audio type '" +
+              type +
+              "' has already existed."
+          );
+          return;
+        }
+
+        this.decoderMap.set(type, decoder);
+      }
+
+      /**
+       * 解码音频数据
+       * @param data 音频文件数据
+       * @param player AudioPlayer实例
+       */
+      static async decodeAudioData(data, player) {
+        // 检查头文件获取音频类型,仅检查前256个字节
+        const toCheck = data.slice(0, 256);
+        const type = checkAudioType(data);
+        if (type === "") {
+          console.error(
+            "Unknown audio type. Header: '" +
+              [...toCheck]
+                .map((v) => v.toString().padStart(2, "0"))
+                .join(" ")
+                .toUpperCase() +
+              "'"
+          );
+          return null;
+        }
+        if (isAudioSupport(type)) {
+          if (data.buffer instanceof ArrayBuffer) {
+            return player.ac.decodeAudioData(data.buffer);
+          } else {
+            return null;
+          }
+        } else {
+          const Decoder = this.decoderMap.get(type);
+          if (!Decoder) {
+            return null;
+          } else {
+            const decoder = new Decoder();
+            await decoder.create();
+            const decodedData = await decoder.decode(data);
+            if (!decodedData) return null;
+            const buffer = player.ac.createBuffer(
+              decodedData.channelData.length,
+              decodedData.channelData[0].length,
+              decodedData.sampleRate
+            );
+            decodedData.channelData.forEach((v, i) => {
+              buffer.copyToChannel(v, i);
+            });
+            decoder.destroy();
+            return buffer;
+          }
         }
       }
-
-      return res;
     }
 
-    const origin = core.events.afterBattle;
-    core.events.afterBattle = function (enemyId, x, y) {
-      // 打怪特效
-      if (has(x) && has(y)) {
-        const frame = core.status.globalAnimateStatus % 2;
-        // 生成怪物图像
-        const canvas = document.createElement("canvas");
-        canvas.width = 32;
-        canvas.height = 32;
-        core.drawIcon(canvas, enemyId, 0, 0, 32, 32, frame);
-        // 执行动画
-        const manager = applyFragWith(canvas);
-        const frag = manager.canvas;
-        // 设置特效画布的css属性
-        frag.style.imageRendering = "pixelated";
-        frag.style.width = `${frag.width * core.domStyle.scale}px`;
-        frag.style.height = `${frag.height * core.domStyle.scale}px`;
-        const left =
-          (x * 32 + 16 - frag.width / 2 - core.bigmap.offsetX) *
-          core.domStyle.scale;
-        const top =
-          (y * 32 + 16 - frag.height / 2 - core.bigmap.offsetY) *
-          core.domStyle.scale;
-        frag.style.left = `${left}px`;
-        frag.style.top = `${top}px`;
-        frag.style.zIndex = "45";
-        frag.style.position = "absolute";
-        // 将特效画布部署到样板上
-        core.dom.gameDraw.appendChild(frag);
-        // 当特效执行完毕后移除这个特效画布
-        manager.onEnd.then(() => {
-          frag.remove();
-        });
+    class VorbisDecoder {
+      /**
+       * 创建音频解码器
+       */
+      async create() {
+        this.decoder = new OggVorbisDecoderWebWorker();
+        await this.decoder.ready;
       }
-      return origin.apply(this, arguments);
+      /**
+       * 摧毁这个解码器
+       */
+      destroy() {
+        this.decoder?.free();
+      }
+      /**
+       * 解码流数据
+       * @param data 流数据
+       */
+
+      async decode(data) {
+        return this.decoder?.decode(data);
+      }
+      /**
+       * 解码整个文件
+       * @param data 文件数据
+       */
+      async decodeAll(data) {
+        return this.decoder?.decodeFile(data);
+      }
+      /**
+       * 当音频解码完成后,会调用此函数,需要返回之前还未解析或未返回的音频数据。调用后,该解码器将不会被再次使用
+       */
+      async flush() {
+        return this.decoder?.flush();
+      }
+    }
+
+    class OpusDecoder {
+      /**
+       * 创建音频解码器
+       */
+      async create() {
+        this.decoder = new OggOpusDecoderWebWorker();
+        await this.decoder.ready;
+      }
+      /**
+       * 摧毁这个解码器
+       */
+      destroy() {
+        this.decoder?.free();
+      }
+      /**
+       * 解码流数据
+       * @param data 流数据
+       */
+      async decode(data) {
+        return this.decoder?.decode(data);
+      }
+      /**
+       * 解码整个文件
+       * @param data 文件数据
+       */
+      async decodeAll(data) {
+        return this.decoder?.decodeFile(data);
+      }
+      /**
+       * 当音频解码完成后,会调用此函数,需要返回之前还未解析或未返回的音频数据。调用后,该解码器将不会被再次使用
+       */
+      async flush() {
+        return await this.decoder?.flush();
+      }
+    }
+    const mimeTypeMap = {
+      [AudioType.Aac]: "audio/aac",
+      [AudioType.Flac]: "audio/flac",
+      [AudioType.Mp3]: "audio/mpeg",
+      [AudioType.Ogg]: "application/ogg",
+      [AudioType.Opus]: "application/ogg",
+      [AudioType.Wav]: "application/ogg",
     };
 
-    if ("frag" in core.plugin) {
-      throw new ReferenceError(`core.plugin上已存在名为frag的属性!`);
+    function isOggPage(data) {
+      return !isNil(data.isFirstPage);
     }
-    core.plugin.frag = {
-      applyFragWith,
+    class AudioStreamSource {
+      constructor(context) {
+        this.output = context.createBufferSource();
+        /** 是否已经完全加载完毕 */
+        this.loaded = false;
+        /** 是否正在播放 */
+        this.playing = false;
+        /** 已经缓冲了多长时间,如果缓冲完那么跟歌曲时长一致 */
+        this.buffered = 0;
+        /** 已经缓冲的采样点数量 */
+        this.bufferedSamples = 0;
+        /** 歌曲时长,加载完毕之前保持为 0 */
+        this.duration = 0;
+        /** 在流传输阶段,至少缓冲多长时间的音频之后才开始播放,单位秒 */
+        this.bufferPlayDuration = 1;
+        /** 音频的采样率,未成功解析出之前保持为 0 */
+        this.sampleRate = 0;
+        //是否循环播放
+        this.loop = false;
+        /** 上一次播放是从何时开始的 */
+        this.lastStartWhen = 0;
+        /** 开始播放时刻 */
+        this.lastStartTime = 0;
+        /** 上一次播放的缓存长度 */
+        this.lastBufferSamples = 0;
+
+        /** 是否已经获取到头文件 */
+        this.headerRecieved = false;
+        /** 音频类型 */
+        this.audioType = "";
+        /** 每多长时间组成一个缓存 Float32Array */
+        this.bufferChunkSize = 10;
+        /** 缓存音频数据,每 bufferChunkSize 秒钟组成一个 Float32Array,用于流式解码 */
+        this.audioData = [];
+
+        this.errored = false;
+        this.ac = context;
+      }
+      /** 当前已经播放了多长时间 */
+      get currentTime() {
+        return this.ac.currentTime - this.lastStartTime + this.lastStartWhen;
+      }
+      /**
+       * 设置每个缓存数据的大小,默认为10秒钟一个缓存数据
+       * @param size 每个缓存数据的时长,单位秒
+       */
+      setChunkSize(size) {
+        if (this.controller?.loading || this.loaded) return;
+        this.bufferChunkSize = size;
+      }
+
+      piped(controller) {
+        this.controller = controller;
+      }
+
+      async pump(data, done) {
+        if (!data || this.errored) return;
+        if (!this.headerRecieved) {
+          // 检查头文件获取音频类型,仅检查前256个字节
+          const toCheck = data.slice(0, 256);
+          this.audioType = checkAudioType(data);
+          if (!this.audioType) {
+            console.error(
+              "Unknown audio type. Header: '" +
+                [...toCheck]
+                  .map((v) => v.toString(16).padStart(2, "0"))
+                  .join(" ")
+                  .toUpperCase() +
+                "'"
+            );
+            return;
+          }
+          // 创建解码器
+          const Decoder = AudioDecoder.decoderMap.get(this.audioType);
+          if (!Decoder) {
+            this.errored = true;
+            console.error(
+              "Cannot decode stream source type of '" +
+                this.audioType +
+                "', since there is no registered decoder for that type."
+            );
+            return Promise.reject(
+              `Cannot decode stream source type of '${this.audioType}', since there is no registered decoder for that type.`
+            );
+          }
+          this.decoder = new Decoder();
+          // 创建数据解析器
+          const mime = mimeTypeMap[this.audioType];
+          const parser = new CodecParser(mime);
+          this.parser = parser;
+          await this.decoder.create();
+          this.headerRecieved = true;
+        }
+
+        const decoder = this.decoder;
+        const parser = this.parser;
+        if (!decoder || !parser) {
+          this.errored = true;
+          return Promise.reject(
+            "No parser or decoder attached in this AudioStreamSource"
+          );
+        }
+
+        await this.decodeData(data, decoder, parser);
+        if (done) await this.decodeFlushData(decoder, parser);
+        this.checkBufferedPlay();
+      }
+
+      /**
+       * 检查采样率,如果还未解析出采样率,那么将设置采样率,如果当前采样率与之前不同,那么发出警告
+       */
+      checkSampleRate(info) {
+        for (const one of info) {
+          const frame = isOggPage(one) ? one.codecFrames[0] : one;
+          if (frame) {
+            const rate = frame.header.sampleRate;
+            if (this.sampleRate === 0) {
+              this.sampleRate = rate;
+              break;
+            } else {
+              if (rate !== this.sampleRate) {
+                console.warn("Sample rate in stream audio must be constant.");
+              }
+            }
+          }
+        }
+      }
+
+      /**
+       * 解析音频数据
+       */
+      async decodeData(data, decoder, parser) {
+        // 解析音频数据
+        const audioData = await decoder.decode(data);
+        if (!audioData) return;
+        // @ts-expect-error 库类型声明错误
+        const audioInfo = [...parser.parseChunk(data)];
+
+        // 检查采样率
+        this.checkSampleRate(audioInfo);
+        // 追加音频数据
+        this.appendDecodedData(audioData, audioInfo);
+      }
+
+      /**
+       * 解码剩余数据
+       */
+      async decodeFlushData(decoder, parser) {
+        const audioData = await decoder.flush();
+        if (!audioData) return;
+        // @ts-expect-error 库类型声明错误
+        const audioInfo = [...parser.flush()];
+
+        this.checkSampleRate(audioInfo);
+        this.appendDecodedData(audioData, audioInfo);
+      }
+
+      /**
+       * 追加音频数据
+       */
+      appendDecodedData(data, info) {
+        const channels = data.channelData.length;
+        if (channels === 0) return;
+        if (this.audioData.length !== channels) {
+          this.audioData = [];
+          for (let i = 0; i < channels; i++) {
+            this.audioData.push([]);
+          }
+        }
+        // 计算出应该放在哪
+        const chunk = this.sampleRate * this.bufferChunkSize;
+        const sampled = this.bufferedSamples;
+        const pushIndex = Math.floor(sampled / chunk);
+        const bufferIndex = sampled % chunk;
+        const dataLength = data.channelData[0].length;
+        let buffered = 0;
+        let nowIndex = pushIndex;
+        let toBuffer = bufferIndex;
+        while (buffered < dataLength) {
+          const rest = toBuffer !== 0 ? chunk - bufferIndex : chunk;
+
+          for (let i = 0; i < channels; i++) {
+            const audioData = this.audioData[i];
+            if (!audioData[nowIndex]) {
+              audioData.push(new Float32Array(chunk));
+            }
+            const toPush = data.channelData[i].slice(buffered, buffered + rest);
+
+            audioData[nowIndex].set(toPush, toBuffer);
+          }
+          buffered += rest;
+          nowIndex++;
+          toBuffer = 0;
+        }
+
+        this.buffered +=
+          info.reduce((prev, curr) => prev + curr.duration, 0) / 1000;
+        this.bufferedSamples += info.reduce(
+          (prev, curr) => prev + curr.samples,
+          0
+        );
+      }
+
+      /**
+       * 检查已缓冲内容,并在未开始播放时播放
+       */
+      checkBufferedPlay() {
+        if (this.playing || this.sampleRate === 0) return;
+        const played = this.lastBufferSamples / this.sampleRate;
+        const dt = this.buffered - played;
+        if (this.loaded) {
+          this.playAudio(played);
+          return;
+        }
+        if (dt < this.bufferPlayDuration) return;
+
+        this.lastBufferSamples = this.bufferedSamples;
+        // 需要播放
+        this.mergeBuffers();
+        if (!this.buffer) return;
+        if (this.playing) this.output.stop();
+        this.createSourceNode(this.buffer);
+        this.output.loop = false;
+        this.output.start(0, played);
+        this.lastStartTime = this.ac.currentTime;
+        this.playing = true;
+        this.output.addEventListener("ended", () => {
+          this.playing = false;
+          this.checkBufferedPlay();
+        });
+      }
+
+      mergeBuffers() {
+        const buffer = this.ac.createBuffer(
+          this.audioData.length,
+          this.bufferedSamples,
+          this.sampleRate
+        );
+        const chunk = this.sampleRate * this.bufferChunkSize;
+        const bufferedChunks = Math.floor(this.bufferedSamples / chunk);
+        const restLength = this.bufferedSamples % chunk;
+        for (let i = 0; i < this.audioData.length; i++) {
+          const audio = this.audioData[i];
+          const data = new Float32Array(this.bufferedSamples);
+          for (let j = 0; j < bufferedChunks; j++) {
+            data.set(audio[j], chunk * j);
+          }
+          if (restLength !== 0) {
+            data.set(
+              audio[bufferedChunks].slice(0, restLength),
+              chunk * bufferedChunks
+            );
+          }
+
+          buffer.copyToChannel(data, i, 0);
+        }
+        this.buffer = buffer;
+      }
+
+      async start() {
+        delete this.buffer;
+        this.headerRecieved = false;
+        this.audioType = "";
+        this.errored = false;
+        this.buffered = 0;
+        this.sampleRate = 0;
+        this.bufferedSamples = 0;
+        this.duration = 0;
+        this.loaded = false;
+        if (this.playing) this.output.stop();
+        this.playing = false;
+        this.lastStartTime = this.ac.currentTime;
+      }
+
+      end(done, reason) {
+        if (done && this.buffer) {
+          this.loaded = true;
+          delete this.controller;
+          this.mergeBuffers();
+
+          this.duration = this.buffered;
+          this.audioData = [];
+          this.decoder?.destroy();
+          delete this.decoder;
+          delete this.parser;
+        } else {
+          console.warn(
+            "Unexpected end when loading stream audio, reason: '" +
+              (reason ?? "") +
+              "'"
+          );
+        }
+      }
+
+      playAudio(when) {
+        if (!this.buffer) return;
+        this.lastStartTime = this.ac.currentTime;
+        if (this.playing) this.output.stop();
+        if (this.route.status !== AudioStatus.Playing) {
+          this.route.status = AudioStatus.Playing;
+        }
+        this.createSourceNode(this.buffer);
+        this.output.start(0, when);
+        this.playing = true;
+
+        this.output.addEventListener("ended", () => {
+          this.playing = false;
+          if (this.route.status === AudioStatus.Playing) {
+            this.route.status = AudioStatus.Stoped;
+          }
+          if (this.loop && !this.output.loop) this.play(0);
+        });
+      }
+      /**
+       * 开始播放这个音频源
+       */
+      play(when) {
+        if (this.playing || this.errored) return;
+        if (this.loaded && this.buffer) {
+          this.playing = true;
+          this.playAudio(when);
+        } else {
+          this.controller?.start();
+        }
+      }
+
+      createSourceNode(buffer) {
+        if (!this.target) return;
+        const node = this.ac.createBufferSource();
+        node.buffer = buffer;
+        if (this.playing) this.output.stop();
+        this.playing = false;
+        this.output = node;
+        node.connect(this.target.input);
+        node.loop = this.loop;
+      }
+      /**
+       * 停止播放这个音频源
+       * @returns 音频暂停的时刻 number
+       */
+      stop() {
+        if (this.playing) this.output.stop();
+        this.playing = false;
+        return this.ac.currentTime - this.lastStartTime;
+      }
+      /**
+       * 连接到音频路由图上,每次调用播放的时候都会执行一次
+       * @param target 连接至的目标 IAudioInput
+       */
+      connect(target) {
+        this.target = target;
+      }
+      /**
+       * 设置是否循环播放
+       * @param loop 是否循环 boolean)
+       */
+      setLoop(loop) {
+        this.loop = loop;
+      }
+    }
+    class AudioElementSource {
+      constructor(context) {
+        const audio = new Audio();
+        audio.preload = "none";
+        this.output = context.createMediaElementSource(audio);
+        this.audio = audio;
+        this.ac = context;
+        audio.addEventListener("play", () => {
+          this.playing = true;
+          if (this.route.status !== AudioStatus.Playing) {
+            this.route.status = AudioStatus.Playing;
+          }
+        });
+        audio.addEventListener("ended", () => {
+          this.playing = false;
+          if (this.route.status === AudioStatus.Playing) {
+            this.route.status = AudioStatus.Stoped;
+          }
+        });
+      }
+      get duration() {
+        return this.audio.duration;
+      }
+      get currentTime() {
+        return this.audio.currentTime;
+      }
+      /**
+       * 设置音频源的路径
+       * @param url 音频路径
+       */
+      setSource(url) {
+        this.audio.src = url;
+      }
+
+      play(when = 0) {
+        if (this.playing) return;
+        this.audio.currentTime = when;
+        this.audio.play();
+      }
+
+      stop() {
+        this.audio.pause();
+        this.playing = false;
+        if (this.route.status === AudioStatus.Playing) {
+          this.route.status = AudioStatus.Stoped;
+        }
+        return this.audio.currentTime;
+      }
+
+      connect(target) {
+        this.output.connect(target.input);
+      }
+
+      setLoop(loop) {
+        this.audio.loop = loop;
+      }
+    }
+    class AudioBufferSource {
+      constructor(context) {
+        this.output = context.createBufferSource();
+        /** 是否循环 */
+        this.loop = false;
+        /** 上一次播放是从何时开始的 */
+        this.lastStartWhen = 0;
+        /** 播放开始时刻 */
+        this.lastStartTime = 0;
+        this.duration = 0;
+        this.ac = context;
+      }
+      get currentTime() {
+        return this.ac.currentTime - this.lastStartTime + this.lastStartWhen;
+      }
+
+      /**
+       * 设置音频源数据
+       * @param buffer 音频源,可以是未解析的 ArrayBuffer,也可以是已解析的 AudioBuffer
+       */
+      async setBuffer(buffer) {
+        if (buffer instanceof ArrayBuffer) {
+          this.buffer = await this.ac.decodeAudioData(buffer);
+        } else {
+          this.buffer = buffer;
+        }
+        this.duration = this.buffer.duration;
+      }
+
+      play(when) {
+        if (this.playing || !this.buffer) return;
+        this.playing = true;
+        this.lastStartTime = this.ac.currentTime;
+        if (this.route.status !== AudioStatus.Playing) {
+          this.route.status = AudioStatus.Playing;
+        }
+        this.createSourceNode(this.buffer);
+        this.output.start(0, when);
+        this.output.addEventListener("ended", () => {
+          this.playing = false;
+          if (this.route.status === AudioStatus.Playing) {
+            this.route.status = AudioStatus.Stoped;
+          }
+          if (this.loop && !this.output.loop) this.play(0);
+        });
+      }
+
+      createSourceNode(buffer) {
+        if (!this.target) return;
+        const node = this.ac.createBufferSource();
+        node.buffer = buffer;
+        this.output = node;
+        node.connect(this.target.input);
+        node.loop = this.loop;
+      }
+
+      stop() {
+        this.output.stop();
+        return this.ac.currentTime - this.lastStartTime;
+      }
+
+      connect(target) {
+        this.target = target;
+      }
+
+      setLoop(loop) {
+        this.loop = loop;
+      }
+    }
+    class AudioPlayer {
+      constructor() {
+        /** 音频播放上下文 */
+        this.ac = new AudioContext();
+        /** 音量节点 */
+        this.gain = this.ac.createGain();
+        this.gain.connect(this.ac.destination);
+        this.audioRoutes = new Map();
+      }
+      /**
+       * 解码音频数据
+       * @param data 音频数据
+       */
+      decodeAudioData(data) {
+        return AudioDecoder.decodeAudioData(data, this);
+      }
+      /**
+       * 设置音量
+       * @param volume 音量
+       */
+      setVolume(volume) {
+        this.gain.gain.value = volume;
+      }
+
+      /**
+       * 获取音量
+       */
+      getVolume() {
+        return this.gain.gain.value;
+      }
+
+      /**
+       * 创建一个音频源
+       * @param Source 音频源类
+       */
+      createSource(Source) {
+        return new Source(this.ac);
+      }
+
+      /**
+       * 创建一个兼容流式音频源,可以与流式加载相结合,主要用于处理 opus ogg 不兼容的情况
+       */
+      createStreamSource() {
+        return new AudioStreamSource(this.ac);
+      }
+
+      /**
+       * 创建一个通过 audio 元素播放的音频源
+       */
+      createElementSource() {
+        return new AudioElementSource(this.ac);
+      }
+
+      /**
+       * 创建一个通过 AudioBuffer 播放的音频源
+       */
+      createBufferSource() {
+        return new AudioBufferSource(this.ac);
+      }
+
+      /**
+       * 获取音频目的地
+       */
+      getDestination() {
+        return this.gain;
+      }
+
+      /**
+       * 创建一个音频效果器
+       * @param Effect 效果器类
+       */
+      createEffect(Effect) {
+        return new Effect(this.ac);
+      }
+
+      /**
+       * 创建一个修改音量的效果器
+       * ```txt
+       *             |----------|
+       * Input ----> | GainNode | ----> Output
+       *             |----------|
+       * ```
+       */
+      createVolumeEffect() {
+        return new VolumeEffect(this.ac);
+      }
+
+      /**
+       * 创建一个立体声效果器
+       * ```txt
+       *             |------------|
+       * Input ----> | PannerNode | ----> Output
+       *             |------------|
+       * ```
+       */
+      createStereoEffect() {
+        return new StereoEffect(this.ac);
+      }
+
+      /**
+       * 创建一个修改单个声道音量的效果器
+       * ```txt
+       *                                  |----------|
+       *                               -> | GainNode | \
+       *             |--------------| /   |----------|  -> |------------|
+       * Input ----> | SplitterNode |        ......        | MergerNode | ----> Output
+       *             |--------------| \   |----------|  -> |------------|
+       *                               -> | GainNode | /
+       *                                  |----------|
+       * ```
+       */
+      createChannelVolumeEffect() {
+        return new ChannelVolumeEffect(this.ac);
+      }
+
+      /**
+       * 创建一个延迟效果器
+       *             |-----------|
+       * Input ----> | DelayNode | ----> Output
+       *             |-----------|
+       */
+      createDelay() {
+        return new DelayEffect(this.ac);
+      }
+
+      /**
+       * 创建一个回声效果器
+       * ```txt
+       *             |----------|
+       * Input ----> | GainNode | ----> Output
+       *        ^    |----------|   |
+       *        |                   |
+       *        |   |------------|  ↓
+       *        |-- | Delay Node | <--
+       *            |------------|
+       * ```
+       */
+      createEchoEffect() {
+        return new EchoEffect(this.ac);
+      }
+
+      /**
+       * 创建一个音频播放路由
+       * @param source 音频源
+       */
+      createRoute(source) {
+        return new AudioRoute(source, this);
+      }
+
+      /**
+       * 添加一个音频播放路由,可以直接被播放
+       * @param id 这个音频播放路由的名称
+       * @param route 音频播放路由对象
+       */
+      addRoute(id, route) {
+        if (!this.audioRoutes) this.audioRoutes = new Map();
+        if (this.audioRoutes.has(id)) {
+          console.warn(
+            "Audio route with id of '" +
+              id +
+              "' has already existed. New route will override old route."
+          );
+        }
+        this.audioRoutes.set(id, route);
+      }
+
+      /**
+       * 根据名称获取音频播放路由对象
+       * @param id 音频播放路由的名称
+       */
+      getRoute(id) {
+        return this.audioRoutes.get(id);
+      }
+      /**
+       * 移除一个音频播放路由
+       * @param id 要移除的播放路由的名称
+       */
+      removeRoute(id) {
+        this.audioRoutes.delete(id);
+      }
+      /**
+       * 播放音频
+       * @param id 音频名称
+       * @param when 从音频的哪个位置开始播放,单位秒
+       */
+      play(id, when) {
+        const route = this.getRoute(id);
+        if (!route) {
+          console.warn(
+            "Cannot play audio route '" +
+              id +
+              "', since there is not added route named it."
+          );
+          return;
+        }
+
+        route.play(when);
+      }
+
+      /**
+       * 暂停音频播放
+       * @param id 音频名称
+       * @returns 当音乐真正停止时兑现
+       */
+      pause(id) {
+        const route = this.getRoute(id);
+        if (!route) {
+          console.warn(
+            "Cannot pause audio route '" +
+              id +
+              "', since there is not added route named it."
+          );
+          return;
+        }
+        return route.pause();
+      }
+
+      /**
+       * 停止音频播放
+       * @param id 音频名称
+       * @returns 当音乐真正停止时兑现
+       */
+      stop(id) {
+        const route = this.getRoute(id);
+        if (!route) {
+          console.warn(
+            "Cannot stop audio route '" +
+              id +
+              "', since there is not added route named it."
+          );
+          return;
+        }
+        return route.stop();
+      }
+
+      /**
+       * 继续音频播放
+       * @param id 音频名称
+       */
+      resume(id) {
+        const route = this.getRoute(id);
+        if (!route) {
+          console.warn(
+            "Cannot pause audio route '" +
+              id +
+              "', since there is not added route named it."
+          );
+          return;
+        }
+        route.resume();
+      }
+
+      /**
+       * 设置听者位置,x正方向水平向右,y正方向垂直于地面向上,z正方向垂直屏幕远离用户
+       * @param x 位置x坐标
+       * @param y 位置y坐标
+       * @param z 位置z坐标
+       */
+      setListenerPosition(x, y, z) {
+        const listener = this.ac.listener;
+        listener.positionX.value = x;
+        listener.positionY.value = y;
+        listener.positionZ.value = z;
+      }
+
+      /**
+       * 设置听者朝向,x正方向水平向右,y正方向垂直于地面向上,z正方向垂直屏幕远离用户
+       * @param x 朝向x坐标
+       * @param y 朝向y坐标
+       * @param z 朝向z坐标
+       */
+      setListenerOrientation(x, y, z) {
+        const listener = this.ac.listener;
+        listener.forwardX.value = x;
+        listener.forwardY.value = y;
+        listener.forwardZ.value = z;
+      }
+
+      /**
+       * 设置听者头顶朝向,x正方向水平向右,y正方向垂直于地面向上,z正方向垂直屏幕远离用户
+       * @param x 头顶朝向x坐标
+       * @param y 头顶朝向y坐标
+       * @param z 头顶朝向z坐标
+       */
+      setListenerUp(x, y, z) {
+        const listener = this.ac.listener;
+        listener.upX.value = x;
+        listener.upY.value = y;
+        listener.upZ.value = z;
+      }
+    }
+    class AudioRoute {
+      constructor(source, player) {
+        source.route = this;
+        this.output = source.output;
+
+        /** 效果器路由图 */
+        this.effectRoute = [];
+
+        /** 结束时长,当音频暂停或停止时,会经过这么长时间之后才真正终止播放,期间可以做音频淡入淡出等效果 */
+        this.endTime = 0;
+        /** 暂停时播放了多长时间 */
+        this.pauseCurrentTime = 0;
+        /** 当前播放状态 */
+        this.player = player;
+        this.status = AudioStatus.Stoped;
+
+        this.shouldStop = false;
+        /**
+         * 每次暂停或停止时自增,用于判断当前正在处理的情况。
+         * 假如暂停后很快播放,然后很快暂停,那么需要根据这个来判断实际是否应该执行暂停后操作
+         */
+        this.stopIdentifier = 0;
+        /** 暂停时刻 */
+        this.pauseTime = 0;
+        this.source = source;
+        this.source.player = player;
+      }
+      /** 音频时长,单位秒 */
+      get duration() {
+        return this.source.duration;
+      }
+      /** 当前播放了多长时间,单位秒 */
+      get currentTime() {
+        if (this.status === AudioStatus.Paused) {
+          return this.pauseCurrentTime;
+        } else {
+          return this.source.currentTime;
+        }
+      }
+      set currentTime(time) {
+        this.source.stop();
+        this.source.play(time);
+      }
+      /**
+       * 设置结束时间,暂停或停止时,会经过这么长时间才终止音频的播放,这期间可以做一下音频淡出的效果。
+       * @param time 暂停或停止时,经过多长时间之后才会结束音频的播放
+       */
+      setEndTime(time) {
+        this.endTime = time;
+      }
+
+      /**
+       * 当音频播放时执行的函数,可以用于音频淡入效果
+       * @param fn 音频开始播放时执行的函数
+       */
+      onStart(fn) {
+        this.audioStartHook = fn;
+      }
+
+      /**
+       * 当音频暂停或停止时执行的函数,可以用于音频淡出效果
+       * @param fn 音频在暂停或停止时执行的函数,不填时表示取消这个钩子。
+       *           包含两个参数,第一个参数是结束时长,第二个参数是当前音频播放路由对象
+       */
+      onEnd(fn) {
+        this.audioEndHook = fn;
+      }
+
+      /**
+       * 开始播放这个音频
+       * @param when 从音频的什么时候开始播放,单位秒
+       */
+      async play(when = 0) {
+        if (this.status === AudioStatus.Playing) return;
+        this.link();
+        await this.player.ac.resume();
+        if (this.effectRoute.length > 0) {
+          const first = this.effectRoute[0];
+          this.source.connect(first);
+          const last = this.effectRoute.at(-1);
+          last.connect({ input: this.player.getDestination() });
+        } else {
+          this.source.connect({ input: this.player.getDestination() });
+        }
+        this.source.play(when);
+        this.status = AudioStatus.Playing;
+        this.pauseTime = 0;
+        this.audioStartHook?.(this);
+        this.startAllEffect();
+        if (this.status !== AudioStatus.Playing) {
+          this.status = AudioStatus.Playing;
+        }
+      }
+
+      /**
+       * 暂停音频播放
+       */
+      async pause() {
+        if (this.status !== AudioStatus.Playing) return;
+        this.status = AudioStatus.Pausing;
+        this.stopIdentifier++;
+        const identifier = this.stopIdentifier;
+        if (this.audioEndHook) {
+          this.audioEndHook(this.endTime, this);
+          await sleep(this.endTime);
+        }
+        if (
+          this.status !== AudioStatus.Pausing ||
+          this.stopIdentifier !== identifier
+        ) {
+          return;
+        }
+        this.pauseCurrentTime = this.source.currentTime;
+        const time = this.source.stop();
+        this.pauseTime = time;
+        if (this.shouldStop) {
+          this.status = AudioStatus.Stoped;
+          this.endAllEffect();
+
+          this.shouldStop = false;
+        } else {
+          this.status = AudioStatus.Paused;
+          this.endAllEffect();
+        }
+        this.endAllEffect();
+      }
+
+      /**
+       * 继续音频播放
+       */
+      resume() {
+        if (this.status === AudioStatus.Playing) return;
+        if (
+          this.status === AudioStatus.Pausing ||
+          this.status === AudioStatus.Stoping
+        ) {
+          this.audioStartHook?.(this);
+
+          return;
+        }
+        if (this.status === AudioStatus.Paused) {
+          this.play(this.pauseTime);
+        } else {
+          this.play(0);
+        }
+        this.status = AudioStatus.Playing;
+        this.pauseTime = 0;
+        this.audioStartHook?.(this);
+        this.startAllEffect();
+      }
+
+      /**
+       * 停止音频播放
+       */
+      async stop() {
+        if (this.status !== AudioStatus.Playing) {
+          if (this.status === AudioStatus.Pausing) {
+            this.shouldStop = true;
+          }
+          return;
+        }
+        this.status = AudioStatus.Stoping;
+        this.stopIdentifier++;
+        const identifier = this.stopIdentifier;
+        if (this.audioEndHook) {
+          this.audioEndHook(this.endTime, this);
+          await sleep(this.endTime);
+        }
+        if (
+          this.status !== AudioStatus.Stoping ||
+          this.stopIdentifier !== identifier
+        ) {
+          return;
+        }
+        this.source.stop();
+        this.status = AudioStatus.Stoped;
+        this.pauseTime = 0;
+        this.endAllEffect();
+      }
+
+      /**
+       * 添加效果器
+       * @param effect 要添加的效果,可以是数组,表示一次添加多个
+       * @param index 从哪个位置开始添加,如果大于数组长度,那么加到末尾,如果小于0,那么将会从后面往前数。默认添加到末尾
+       */
+      addEffect(effect, index) {
+        if (isNil(index)) {
+          if (effect instanceof Array) {
+            this.effectRoute.push(...effect);
+          } else {
+            this.effectRoute.push(effect);
+          }
+        } else {
+          if (effect instanceof Array) {
+            this.effectRoute.splice(index, 0, ...effect);
+          } else {
+            this.effectRoute.splice(index, 0, effect);
+          }
+        }
+        this.setOutput();
+        if (this.source.playing) this.link();
+      }
+
+      /**
+       * 移除一个效果器
+       * @param effect 要移除的效果
+       */
+      removeEffect(effect) {
+        const index = this.effectRoute.indexOf(effect);
+        if (index === -1) return;
+        this.effectRoute.splice(index, 1);
+        effect.disconnect();
+        this.setOutput();
+        if (this.source.playing) this.link();
+      }
+
+      setOutput() {
+        const effect = this.effectRoute.at(-1);
+        if (!effect) this.output = this.source.output;
+        else this.output = effect.output;
+      }
+
+      /**
+       * 连接音频路由图
+       */
+      link() {
+        this.effectRoute.forEach((v) => v.disconnect());
+        this.effectRoute.forEach((v, i) => {
+          const next = this.effectRoute[i + 1];
+          if (next) {
+            v.connect(next);
+          }
+        });
+      }
+
+      startAllEffect() {
+        this.effectRoute.forEach((v) => v.start());
+      }
+
+      endAllEffect() {
+        this.effectRoute.forEach((v) => v.end());
+      }
+    }
+
+    const audioPlayer = new AudioPlayer();
+
+    class BgmController {
+      constructor(player) {
+        this.mainGain = player.createVolumeEffect();
+        this.player = player;
+        /** bgm音频名称的前缀 */
+        this.prefix = "bgms.";
+        /** 每个 bgm 的音量控制器 */
+        this.gain = new Map();
+
+        /** 正在播放的 bgm */
+        this.playingBgm = "";
+        /** 是否正在播放 */
+        this.playing = false;
+
+        /** 是否已经启用 */
+        this.enabled = true;
+        /** 是否屏蔽所有的音乐切换 */
+        this.blocking = false;
+        /** 渐变时长 */
+        this.transitionTime = 2000;
+      }
+
+      /**
+       * 设置音频渐变时长
+       * @param time 渐变时长
+       */
+      setTransitionTime(time) {
+        this.transitionTime = time;
+        for (const [, value] of this.gain) {
+          value.transition.time(time);
+        }
+      }
+
+      /**
+       * 屏蔽音乐切换
+       */
+      blockChange() {
+        this.blocking = true;
+      }
+
+      /**
+       * 取消屏蔽音乐切换
+       */
+      unblockChange() {
+        this.blocking = false;
+      }
+
+      /**
+       * 设置总音量大小
+       * @param volume 音量大小
+       */
+      setVolume(volume) {
+        this.mainGain.setVolume(volume);
+        this._volume = volume;
+      }
+      /**
+       * 获取总音量大小
+       */
+      getVolume() {
+        return this.mainGain.getVolume();
+      }
+      /**
+       * 设置是否启用
+       * @param enabled 是否启用
+       */
+      setEnabled(enabled) {
+        if (enabled) this.resume();
+        else this.stop();
+        this.enabled = enabled;
+      }
+
+      /**
+       * 设置 bgm 音频名称的前缀
+       */
+      setPrefix(prefix) {
+        this.prefix = prefix;
+      }
+
+      getId(name) {
+        return `${this.prefix}${name}`;
+      }
+
+      /**
+       * 根据 bgm 名称获取其 AudioRoute 实例
+       * @param id 音频名称
+       */
+      get(id) {
+        return this.player.getRoute(this.getId(id));
+      }
+
+      /**
+       * 添加一个 bgm
+       * @param id 要添加的 bgm 的名称
+       * @param url 指定 bgm 的加载地址
+       */
+      addBgm(id, url = `project/bgms/${id}`) {
+        const type = guessTypeByExt(id);
+        if (!type) {
+          console.warn(
+            "Unknown audio extension name: '" +
+              id.split(".").slice(0, -1).join(".") +
+              "'"
+          );
+          return;
+        }
+        const gain = this.player.createVolumeEffect();
+        if (isAudioSupport(type)) {
+          const source = audioPlayer.createElementSource();
+          source.setSource(url);
+          source.setLoop(true);
+          const route = new AudioRoute(source, audioPlayer);
+          route.addEffect([gain, this.mainGain]);
+          audioPlayer.addRoute(this.getId(id), route);
+          this.setTransition(id, route, gain);
+        } else {
+          const source = audioPlayer.createStreamSource();
+          const stream = new StreamLoader(url);
+          stream.pipe(source);
+          source.setLoop(true);
+          const route = new AudioRoute(source, audioPlayer);
+          route.addEffect([gain, this.mainGain]);
+          audioPlayer.addRoute(this.getId(id), route);
+          this.setTransition(id, route, gain);
+        }
+      }
+
+      /**
+       * 移除一个 bgm
+       * @param id 要移除的 bgm 的名称
+       */
+      removeBgm(id) {
+        this.player.removeRoute(this.getId(id));
+        const gain = this.gain.get(id);
+        gain?.transition.ticker.destroy();
+        this.gain.delete(id);
+      }
+
+      setTransition(id, route, gain) {
+        const transition = new Transition();
+        transition
+          .time(this.transitionTime)
+          .mode(linear())
+          .transition("volume", 0);
+
+        const tick = () => {
+          gain.setVolume(transition.value.volume);
+        };
+
+        /**
+         * @param expect 在结束时应该是正在播放还是停止
+         */
+        const setTick = async (expect) => {
+          transition.ticker.remove(tick);
+          transition.ticker.add(tick);
+          const identifier = route.stopIdentifier;
+          await sleep(this.transitionTime + 500);
+          if (route.status === expect && identifier === route.stopIdentifier) {
+            transition.ticker.remove(tick);
+            if (route.status === AudioStatus.Playing) {
+              gain.setVolume(1);
+            } else {
+              gain.setVolume(0);
+            }
+          }
+        };
+
+        route.onStart(async () => {
+          transition.transition("volume", 1);
+          setTick(AudioStatus.Playing);
+        });
+        route.onEnd(() => {
+          transition.transition("volume", 0);
+          setTick(AudioStatus.Paused);
+        });
+        route.setEndTime(this.transitionTime);
+
+        this.gain.set(id, { effect: gain, transition });
+      }
+
+      /**
+       * 播放一个 bgm
+       * @param id 要播放的 bgm 名称
+       */
+      play(id, when) {
+        if (this.blocking) return;
+        if (id !== this.playingBgm && this.playingBgm) {
+          this.player.pause(this.getId(this.playingBgm));
+        }
+        this.playingBgm = id;
+        if (!this.enabled) return;
+        this.player.play(this.getId(id), when);
+        this.playing = true;
+      }
+
+      /**
+       * 继续当前的 bgm
+       */
+      resume() {
+        if (this.blocking || !this.enabled || this.playing) return;
+        if (this.playingBgm) {
+          this.player.resume(this.getId(this.playingBgm));
+        }
+        this.playing = true;
+      }
+
+      /**
+       * 暂停当前的 bgm
+       */
+      pause() {
+        if (this.blocking || !this.enabled) return;
+        if (this.playingBgm) {
+          this.player.pause(this.getId(this.playingBgm));
+        }
+        this.playing = false;
+      }
+
+      /**
+       * 停止当前的 bgm
+       */
+      stop() {
+        if (this.blocking || !this.enabled) return;
+        if (this.playingBgm) {
+          this.player.stop(this.getId(this.playingBgm));
+        }
+        this.playing = false;
+      }
+    }
+    const bgmController = new BgmController(audioPlayer);
+
+    class SoundPlayer {
+      constructor(player) {
+        /** 每个音效的唯一标识符 */
+        this.num = 0;
+        this.enabled = true;
+        this.gain = player.createVolumeEffect();
+        /** 每个音效的数据 */
+        this.buffer = new Map();
+        /** 所有正在播放的音乐 */
+        this.playing = new Set();
+        this.player = player;
+      }
+      /**
+       * 设置是否启用音效
+       * @param enabled 是否启用音效
+       */
+      setEnabled(enabled) {
+        if (!enabled) this.stopAllSounds();
+        this.enabled = enabled;
+      }
+
+      /**
+       * 设置音量大小
+       * @param volume 音量大小
+       */
+      setVolume(volume) {
+        this.gain.setVolume(volume);
+      }
+      /**
+       * 获取音量大小
+       */
+      getVolume() {
+        return this.gain.getVolume();
+      }
+      /**
+       * 添加一个音效
+       * @param id 音效名称
+       * @param data 音效的Uint8Array数据
+       */
+      async add(id, data) {
+        const buffer = await this.player.decodeAudioData(data);
+        if (!buffer) {
+          console.warn(
+            "Cannot decode sound '" +
+              id +
+              "', since audio file may not supported by 2.b."
+          );
+          return;
+        }
+        this.buffer.set(id, buffer);
+      }
+
+      /**
+       * 播放一个音效
+       * @param id 音效名称
+       * @param position 音频位置,[0, 0, 0]表示正中心,x轴指向水平向右,y轴指向水平向上,z轴指向竖直向上
+       * @param orientation 音频朝向,[0, 1, 0]表示朝向前方
+       */
+      play(id, position = [0, 0, 0], orientation = [1, 0, 0]) {
+        if (!this.enabled || !id) return -1;
+        const buffer = this.buffer.get(id);
+        if (!buffer) {
+          console.warn(
+            "Cannot play sound '" +
+              id +
+              "', since there is no added data named it."
+          );
+          return -1;
+        }
+        const soundNum = this.num++;
+
+        const source = this.player.createBufferSource();
+        source.setBuffer(buffer);
+        const route = this.player.createRoute(source);
+        const stereo = this.player.createStereoEffect();
+        stereo.setPosition(position[0], position[1], position[2]);
+        stereo.setOrientation(orientation[0], orientation[1], orientation[2]);
+        route.addEffect([stereo, this.gain]);
+        this.player.addRoute(`sounds.${soundNum}`, route);
+        route.play();
+        source.output.addEventListener("ended", () => {
+          this.playing.delete(soundNum);
+        });
+        this.playing.add(soundNum);
+        return soundNum;
+      }
+
+      /**
+       * 停止一个音效
+       * @param num 音效的唯一 id
+       */
+      stop(num) {
+        const id = `sounds.${num}`;
+        const route = this.player.getRoute(id);
+        if (route) {
+          route.stop();
+          this.player.removeRoute(id);
+          this.playing.delete(num);
+        }
+      }
+
+      /**
+       * 停止播放所有音效
+       */
+      stopAllSounds() {
+        this.playing.forEach((v) => {
+          const id = `sounds.${v}`;
+          const route = this.player.getRoute(id);
+          if (route) {
+            route.stop();
+            this.player.removeRoute(id);
+          }
+        });
+        this.playing.clear();
+      }
+    }
+    const soundPlayer = new SoundPlayer(audioPlayer);
+
+    function loadAllBgm() {
+      const data = data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d;
+      for (const bgm of data.main.bgms) {
+        bgmController.addBgm(bgm);
+      }
+    }
+    loadAllBgm();
+    AudioDecoder.registerDecoder(AudioType.Ogg, VorbisDecoder);
+    AudioDecoder.registerDecoder(AudioType.Opus, OpusDecoder);
+
+    core.plugin.audioSystem = {
+      AudioType,
+      AudioDecoder,
+      AudioStatus,
+      checkAudioType,
+      isAudioSupport,
+      audioPlayer,
+      soundPlayer,
+      bgmController,
+      guessTypeByExt,
+      BgmController,
+      SoundPlayer,
+      EchoEffect,
+      DelayEffect,
+      ChannelVolumeEffect,
+      VolumeEffect,
+      StereoEffect,
+      AudioEffect,
+      AudioPlayer,
+      AudioRoute,
+      AudioStreamSource,
+      AudioElementSource,
+      AudioBufferSource,
+      loadAllBgm,
+      StreamLoader,
+    };
+    //bgm相关复写
+    control.prototype.playBgm = (bgm, when) => {
+      bgm = core.getMappedName(bgm);
+      bgmController.play(bgm, when);
+      core.setMusicBtn();
+    };
+    control.prototype.pauseBgm = () => {
+      bgmController.pause();
+      core.setMusicBtn();
+    };
+
+    control.prototype.resumeBgm = function () {
+      bgmController.resume();
+      core.setMusicBtn();
+    };
+    control.prototype.checkBgm = function () {
+      core.playBgm(bgmController.playingBgm || main.startBgm);
+    };
+    control.prototype.triggerBgm = function () {
+      core.musicStatus.bgmStatus = !core.musicStatus.bgmStatus;
+      if (bgmController.playing) bgmController.pause();
+      else bgmController.resume();
+      core.setMusicBtn();
+      core.setLocalStorage("bgmStatus", core.musicStatus.bgmStatus);
+    };
+    //sound相关复写
+    control.prototype.playSound = function (
+      sound,
+      _pitch,
+      callback,
+      position,
+      orientation
+    ) {
+      if (main.mode != "play" || !core.musicStatus.soundStatus) return;
+      const name = core.getMappedName(sound);
+      const num = soundPlayer.play(name, position, orientation);
+      const route = audioPlayer.getRoute(`sounds.${num}`);
+      if (!route) {
+        callback?.();
+        return -1;
+      } else {
+        sleep(route.duration * 1000).then(() => callback?.());
+        return num;
+      }
+    };
+    control.prototype.stopSound = function (id) {
+      if (isNil(id)) {
+        soundPlayer.stopAllSounds();
+      } else {
+        soundPlayer.stop(id);
+      }
+    };
+    control.prototype.getPlayingSounds = function () {
+      return [...soundPlayer.playing];
+    };
+    //sound加载复写
+    loader.prototype._loadOneSound_decodeData = function (name, data) {
+      if (data instanceof Blob) {
+        var blobReader = new zip.BlobReader(data);
+        blobReader.init(function () {
+          blobReader.readUint8Array(0, blobReader.size, function (uint8) {
+            //core.loader._loadOneSound_decodeData(name, uint8.buffer);
+            soundPlayer.add(name, uint8);
+          });
+        });
+        return;
+      }
+      if (data instanceof ArrayBuffer) {
+        const uint8 = new Uint8Array(data);
+        soundPlayer.add(name, uint8);
+      }
+    };
+    //音量控制复写
+    soundPlayer.setVolume(
+      core.musicStatus.userVolume * core.musicStatus.designVolume
+    );
+    bgmController.setVolume(
+      core.musicStatus.userVolume * core.musicStatus.designVolume
+    );
+    actions.prototype._clickSwitchs_sounds_userVolume = function (delta) {
+      var value = Math.round(Math.sqrt(100 * core.musicStatus.userVolume));
+      if (value == 0 && delta < 0) return;
+      core.musicStatus.userVolume = core.clamp(
+        Math.pow(value + delta, 2) / 100,
+        0,
+        1
+      );
+      //audioContext 音效 不受designVolume 影响
+      if (core.musicStatus.gainNode != null)
+        core.musicStatus.gainNode.gain.value = core.musicStatus.userVolume;
+      soundPlayer.setVolume(
+        core.musicStatus.userVolume * core.musicStatus.designVolume
+      );
+      bgmController.setVolume(
+        core.musicStatus.userVolume * core.musicStatus.designVolume
+      );
+      core.setLocalStorage("userVolume", core.musicStatus.userVolume);
+      core.playSound("确定");
+      core.ui._drawSwitchs_sounds();
     };
   },
     "自定义常用事件": function () {
-    // editorBlocklyconfigPlus.js
-    // 自訂常見事件模板插件
-    // 本插件引用了通用函數插件(Utility.js)
-    // 適用樣板:2.10.3
-    // 請注意:
-    // 此插件對事件編輯器(editor_blocklyconfig)進行複寫,若還有其它針對事件編輯器做複寫的插件,請謹慎使用!
-    // 此插件對表格操作行為(editor_mode.doActionList)進行複寫,若還有其它對表格操作行為做複寫的插件,請謹慎使用!
-    // 使用方法:
-    // 現在在主頁下拉選單多了個常用事件模版,在那邊可以自由設定常用事件模板。
-    // 設定完後按F5刷新,再到事件編輯器看就有你設定好的常用事件模板了。
+	// editorBlocklyconfigPlus.js
+	// 自訂常見事件模板插件
+	// 本插件引用了通用函數插件(Utility.js)
+	// 適用樣板:2.10.3
+	// 請注意:
+	// 此插件對事件編輯器(editor_blocklyconfig)進行複寫,若還有其它針對事件編輯器做複寫的插件,請謹慎使用!
+	// 此插件對表格操作行為(editor_mode.doActionList)進行複寫,若還有其它對表格操作行為做複寫的插件,請謹慎使用!
+	// 使用方法:
+	// 現在在主頁下拉選單多了個常用事件模版,在那邊可以自由設定常用事件模板。
+	// 設定完後按F5刷新,再到事件編輯器看就有你設定好的常用事件模板了。
 
-    if (main.mode == "editor") {
-      //#region 配置表格初始化
-      let TableFileName = "project/table/CommonEventTemplate_comment.js";
-      let TableRow = `
+	if (main.mode == "editor") {
+		//#region 配置表格初始化
+		let TableFileName = "project/table/CommonEventTemplate_comment.js";
+		let TableRow = `
 				var CommonEventTemplate_comment = {"_type": "object",
 				"_data": {
 					"CommonEventTemplate": {
@@ -10851,241 +11696,221 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
 					}
 				}}
 			`;
-      if (!events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate) {
-        /**
-         * @type {{[EvnetName:actionParserJson]}}
-         */
-        events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate = {
-          检测音乐如果没有开启则系统提示开启: [
-            {
-              type: "if",
-              condition: "!core.musicStatus.bgmStatus",
-              true: [
-                "\t[系统提示]你当前音乐处于关闭状态,本塔开音乐游戏效果更佳",
-              ],
-              false: [],
-            },
-          ],
-          仿新新魔塔一次性商人: [
-            {
-              type: "if",
-              condition: "switch:A",
-              true: [
-                "\t[行商,trader]\b[this]这是购买我的道具后我给玩家的提示。",
-                {
-                  type: "comment",
-                  text: "下一条指令可视情况使用或不使用",
-                },
-                {
-                  type: "hide",
-                  remove: true,
-                  time: 250,
-                },
-              ],
-              false: [
-                {
-                  type: "confirm",
-                  text: "我有3把黄钥匙,\n你出50金币就卖给你。",
-                  yes: [
-                    {
-                      type: "if",
-                      condition: "status:money>=50",
-                      true: [
-                        {
-                          type: "setValue",
-                          name: "status:money",
-                          operator: "-=",
-                          value: "50",
-                        },
-                        {
-                          type: "setValue",
-                          name: "item:yellowKey",
-                          operator: "+=",
-                          value: "3",
-                        },
-                        {
-                          type: "playSound",
-                          name: "确定",
-                          stop: true,
-                        },
-                        {
-                          type: "setValue",
-                          name: "switch:A",
-                          value: "true",
-                        },
-                      ],
-                      false: [
-                        {
-                          type: "playSound",
-                          name: "操作失败",
-                        },
-                        "\t[行商,trader]\b[this]你的金币不足!",
-                      ],
-                    },
-                  ],
-                  no: [],
-                },
-              ],
-            },
-          ],
-          全地图选中一个点: [
-            {
-              type: "comment",
-              text: "全地图选中一个点,需要用鼠标或触屏操作",
-            },
-            {
-              type: "setValue",
-              name: "temp:X",
-              value: "status:x",
-            },
-            {
-              type: "setValue",
-              name: "temp:Y",
-              value: "status:y",
-            },
-            {
-              type: "tip",
-              text: "再次点击闪烁位置确认",
-            },
-            {
-              type: "while",
-              condition: "true",
-              data: [
-                {
-                  type: "drawSelector",
-                  image: "winskin.webp",
-                  code: 1,
-                  x: "32*temp:X",
-                  y: "32*temp:Y",
-                  width: 32,
-                  height: 32,
-                },
-                {
-                  type: "wait",
-                },
-                {
-                  type: "if",
-                  condition: "(flag:type === 1)",
-                  true: [
-                    {
-                      type: "if",
-                      condition: "((temp:X===flag:x)&&(temp:Y===flag:y))",
-                      true: [
-                        {
-                          type: "break",
-                          n: 1,
-                        },
-                      ],
-                    },
-                    {
-                      type: "setValue",
-                      name: "temp:X",
-                      value: "flag:x",
-                    },
-                    {
-                      type: "setValue",
-                      name: "temp:Y",
-                      value: "flag:y",
-                    },
-                  ],
-                },
-              ],
-            },
-            {
-              type: "drawSelector",
-              code: 1,
-            },
-            {
-              type: "comment",
-              text: "流程进行到这里可以对[X,Y]点进行处理,比如",
-            },
-            {
-              type: "closeDoor",
-              id: "yellowDoor",
-              loc: ["temp:X", "temp:Y"],
-            },
-          ],
-          多阶段Boss战斗: [
-            {
-              type: "comment",
-              text: "多阶段boss,请直接作为战后事件使用",
-            },
-            {
-              type: "setValue",
-              name: "switch:A",
-              operator: "+=",
-              value: "1",
-            },
-            {
-              type: "switch",
-              condition: "switch:A",
-              caseList: [
-                {
-                  case: "1",
-                  action: [
-                    {
-                      type: "setBlock",
-                      number: "redSlime",
-                    },
-                    "\t[2阶段boss,redSlime]\b[this]你以为你已经打败我了吗?没听说过史莱姆有九条命吗?",
-                  ],
-                },
-                {
-                  case: "2",
-                  action: [
-                    {
-                      type: "setBlock",
-                      number: "blackSlime",
-                    },
-                    "\t[3阶段boss,blackSlime]\b[this]不能消灭我的,只会让我更强大!",
-                  ],
-                },
-                {
-                  case: "3",
-                  action: [
-                    {
-                      type: "setBlock",
-                      number: "slimelord",
-                    },
-                    "\t[4阶段boss,slimelord]\b[this]我还能打!",
-                  ],
-                },
-                {
-                  case: "4",
-                  action: ["\t[4阶段boss,slimelord]我一定会回来的!"],
-                },
-              ],
-            },
-          ],
-        };
-      }
-      //#endregion
+		if (!events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate) {
+			/**
+			 * @type {{[EvnetName:actionParserJson]}}
+			 */
+			events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate = {
+				检测音乐如果没有开启则系统提示开启: [{
+					type: "if",
+					condition: "!core.musicStatus.bgmStatus",
+					true: [
+						"\t[系统提示]你当前音乐处于关闭状态,本塔开音乐游戏效果更佳",
+					],
+					false: [],
+				}, ],
+				仿新新魔塔一次性商人: [{
+					type: "if",
+					condition: "switch:A",
+					true: [
+						"\t[行商,trader]\b[this]这是购买我的道具后我给玩家的提示。",
+						{
+							type: "comment",
+							text: "下一条指令可视情况使用或不使用",
+						},
+						{
+							type: "hide",
+							remove: true,
+							time: 250,
+						},
+					],
+					false: [{
+						type: "confirm",
+						text: "我有3把黄钥匙,\n你出50金币就卖给你。",
+						yes: [{
+							type: "if",
+							condition: "status:money>=50",
+							true: [{
+									type: "setValue",
+									name: "status:money",
+									operator: "-=",
+									value: "50",
+								},
+								{
+									type: "setValue",
+									name: "item:yellowKey",
+									operator: "+=",
+									value: "3",
+								},
+								{
+									type: "playSound",
+									name: "确定",
+									stop: true,
+								},
+								{
+									type: "setValue",
+									name: "switch:A",
+									value: "true",
+								},
+							],
+							false: [{
+									type: "playSound",
+									name: "操作失败",
+								},
+								"\t[行商,trader]\b[this]你的金币不足!",
+							],
+						}, ],
+						no: [],
+					}, ],
+				}, ],
+				全地图选中一个点: [{
+						type: "comment",
+						text: "全地图选中一个点,需要用鼠标或触屏操作",
+					},
+					{
+						type: "setValue",
+						name: "temp:X",
+						value: "status:x",
+					},
+					{
+						type: "setValue",
+						name: "temp:Y",
+						value: "status:y",
+					},
+					{
+						type: "tip",
+						text: "再次点击闪烁位置确认",
+					},
+					{
+						type: "while",
+						condition: "true",
+						data: [{
+								type: "drawSelector",
+								image: "winskin.webp",
+								code: 1,
+								x: "32*temp:X",
+								y: "32*temp:Y",
+								width: 32,
+								height: 32,
+							},
+							{
+								type: "wait",
+							},
+							{
+								type: "if",
+								condition: "(flag:type === 1)",
+								true: [{
+										type: "if",
+										condition: "((temp:X===flag:x)&&(temp:Y===flag:y))",
+										true: [{
+											type: "break",
+											n: 1,
+										}, ],
+									},
+									{
+										type: "setValue",
+										name: "temp:X",
+										value: "flag:x",
+									},
+									{
+										type: "setValue",
+										name: "temp:Y",
+										value: "flag:y",
+									},
+								],
+							},
+						],
+					},
+					{
+						type: "drawSelector",
+						code: 1,
+					},
+					{
+						type: "comment",
+						text: "流程进行到这里可以对[X,Y]点进行处理,比如",
+					},
+					{
+						type: "closeDoor",
+						id: "yellowDoor",
+						loc: ["temp:X", "temp:Y"],
+					},
+				],
+				多阶段Boss战斗: [{
+						type: "comment",
+						text: "多阶段boss,请直接作为战后事件使用",
+					},
+					{
+						type: "setValue",
+						name: "switch:A",
+						operator: "+=",
+						value: "1",
+					},
+					{
+						type: "switch",
+						condition: "switch:A",
+						caseList: [{
+								case: "1",
+								action: [{
+										type: "setBlock",
+										number: "redSlime",
+									},
+									"\t[2阶段boss,redSlime]\b[this]你以为你已经打败我了吗?没听说过史莱姆有九条命吗?",
+								],
+							},
+							{
+								case: "2",
+								action: [{
+										type: "setBlock",
+										number: "blackSlime",
+									},
+									"\t[3阶段boss,blackSlime]\b[this]不能消灭我的,只会让我更强大!",
+								],
+							},
+							{
+								case: "3",
+								action: [{
+										type: "setBlock",
+										number: "slimelord",
+									},
+									"\t[4阶段boss,slimelord]\b[this]我还能打!",
+								],
+							},
+							{
+								case: "4",
+								action: ["\t[4阶段boss,slimelord]我一定会回来的!"],
+							},
+						],
+					},
+				],
+			};
+		}
+		//#endregion
 
-      // 新增模板選項
-      let editModeSelect = document.getElementById("editModeSelect");
-      let newEditModeOption = document.createElement("option");
-      newEditModeOption.value = "CommonEventTemplate";
-      newEditModeOption.text = "常見事件模板";
-      editModeSelect.add(newEditModeOption);
+		// 新增模板選項
+		let editModeSelect = document.getElementById("editModeSelect");
+		let newEditModeOption = document.createElement("option");
+		newEditModeOption.value = "CommonEventTemplate";
+		newEditModeOption.text = "常見事件模板";
+		editModeSelect.add(newEditModeOption);
 
-      //檢查可用的編輯模板ID
-      let leftIDNumber = 11 - 1;
-      let ExistLeftElement = document.querySelector(".main");
-      while (ExistLeftElement) {
-        leftIDNumber++;
-        ExistLeftElement = document.getElementById(`left${leftIDNumber}`);
-      }
+		//檢查可用的編輯模板ID
+		let leftIDNumber = 11 - 1;
+		let ExistLeftElement = document.querySelector(".main");
+		while (ExistLeftElement) {
+			leftIDNumber++;
+			ExistLeftElement = document.getElementById(`left${leftIDNumber}`);
+		}
 
-      //新增編輯模板
-      let MainDiv = document.querySelector(".main");
+		//新增編輯模板
+		let MainDiv = document.querySelector(".main");
 
-      let CommonEventTemplateMainDiv = document.createElement("div");
-      CommonEventTemplateMainDiv.id = `left${leftIDNumber}`;
-      CommonEventTemplateMainDiv.className = "leftTab";
-      CommonEventTemplateMainDiv.style.zIndex = "-1";
-      CommonEventTemplateMainDiv.style.opacity = "0";
+		let CommonEventTemplateMainDiv = document.createElement("div");
+		CommonEventTemplateMainDiv.id = `left${leftIDNumber}`;
+		CommonEventTemplateMainDiv.className = "leftTab";
+		CommonEventTemplateMainDiv.style.zIndex = "-1";
+		CommonEventTemplateMainDiv.style.opacity = "0";
 
-      CommonEventTemplateMainDiv.innerHTML = `
+		CommonEventTemplateMainDiv.innerHTML = `
 				<!-- CommonEventTemplate -->
 				<h3 class="leftTabHeader">
 				常見事件模板  
@@ -11108,980 +11933,968 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
 					</div>
 				</div>
 			`;
-      MainDiv.appendChild(CommonEventTemplateMainDiv);
+		MainDiv.appendChild(CommonEventTemplateMainDiv);
 
-      (async function () {
-        //等待編輯器初始化
-        while (!editor_mode.ids) {
-          await Sleep(100);
-        }
-        //新增編輯模板ID
-        editor_mode.ids["CommonEventTemplate"] = `left${leftIDNumber}`;
-        editor_mode.init_dom_ids();
-        //切換至常見事件模板
-        editor_mode.CommonEventTemplate = function (callback) {
-          var objs = [];
-          editor.file.editCommonEventTemplate([], function (objs_) {
-            objs = objs_;
-            //console.log(objs_)
-          });
-          //只查询不修改时,内部实现不是异步的,所以可以这么写
-          var tableinfo = editor.table.objToTable(objs[0], objs[1]);
-          document.getElementById(
-            "table_298572d8-93dd-4c6e-a278-6a7d49831e3a"
-          ).innerHTML = tableinfo.HTML;
-          tableinfo.listen(tableinfo.guids);
-          if (Boolean(callback)) callback();
-        };
+		(async function () {
+			//等待編輯器初始化
+			while (!editor_mode.ids) {
+				await Sleep(100);
+			}
+			//新增編輯模板ID
+			editor_mode.ids["CommonEventTemplate"] = `left${leftIDNumber}`;
+			editor_mode.init_dom_ids();
+			//切換至常見事件模板
+			editor_mode.CommonEventTemplate = function (callback) {
+				var objs = [];
+				editor.file.editCommonEventTemplate([], function (objs_) {
+					objs = objs_;
+					//console.log(objs_)
+				});
+				//只查询不修改时,内部实现不是异步的,所以可以这么写
+				var tableinfo = editor.table.objToTable(objs[0], objs[1]);
+				document.getElementById(
+					"table_298572d8-93dd-4c6e-a278-6a7d49831e3a"
+				).innerHTML = tableinfo.HTML;
+				tableinfo.listen(tableinfo.guids);
+				if (Boolean(callback)) callback();
+			};
 
-        //檢查配置表格存在
-        let TableRowExist = null;
-        fs.readFile(TableFileName, "base64", function (err, data) {
-          if (err) {
-            console.log(`察覺常見事件模板配置表格不存在,原因:${err}`);
-            console.log("新建一個常見事件模板配置表格。");
-            TableRowExist = false;
-          } else {
-            TableRowExist = true;
-          }
-        });
-        //等待配置表格載入完畢(最多0.3秒,超過則視為失敗)
-        for (let i = 0; i < 3; i++) {
-          if (TableRowExist == null) {
-            await Sleep(100);
-          }
-        }
-        //配置表格初始化
-        if (TableRowExist != true) {
-          fs.mkdir("project/table", function (err, data) {
-            if (err) throw `常見事件模板配置表格目錄初始化失敗,原因:${err}`;
-          });
-          fs.writeFile(
-            TableFileName,
-            editor.util.encode64(TableRow || ""),
-            "base64",
-            function (err, data) {
-              if (err) throw `常見事件模板配置表格文件初始化失敗,原因:${err}`;
-            }
-          );
-        }
-        //載入配置表格
-        //editor.file.loadCommentjs(callback);
-        (function () {
-          var key = "CommonEventTemplate_comment";
-          var script = document.createElement("script");
-          script.src = "project/table/" + key + ".js";
-          document.body.appendChild(script);
-          script.onload = function () {
-            editor.file[key] = eval(key.replace(".", "_"));
-            var loaded = Boolean(editor.file[key]);
-          };
-        })();
-        //按下配置表格
-        editor_multi.CommonEventTemplateEditCommentJs = function (mod) {
-          editor_multi.lintAutocomplete = true;
-          editor_multi.setLint();
-          editor_multi.importFile(TableFileName);
-        };
+			//檢查配置表格存在
+			let TableRowExist = null;
+			fs.readFile(TableFileName, "base64", function (err, data) {
+				if (err) {
+					console.log(`察覺常見事件模板配置表格不存在,原因:${err}`);
+					console.log("新建一個常見事件模板配置表格。");
+					TableRowExist = false;
+				} else {
+					TableRowExist = true;
+				}
+			});
+			//等待配置表格載入完畢(最多0.3秒,超過則視為失敗)
+			for (let i = 0; i < 3; i++) {
+				if (TableRowExist == null) {
+					await Sleep(100);
+				}
+			}
+			//配置表格初始化
+			if (TableRowExist != true) {
+				fs.mkdir("project/table", function (err, data) {
+					if (err) throw `常見事件模板配置表格目錄初始化失敗,原因:${err}`;
+				});
+				fs.writeFile(
+					TableFileName,
+					editor.util.encode64(TableRow || ""),
+					"base64",
+					function (err, data) {
+						if (err) throw `常見事件模板配置表格文件初始化失敗,原因:${err}`;
+					}
+				);
+			}
+			//載入配置表格
+			//editor.file.loadCommentjs(callback);
+			(function () {
+				var key = "CommonEventTemplate_comment";
+				var script = document.createElement("script");
+				script.src = "project/table/" + key + ".js";
+				document.body.appendChild(script);
+				script.onload = function () {
+					editor.file[key] = eval(key.replace(".", "_"));
+					var loaded = Boolean(editor.file[key]);
+				};
+			})();
+			//按下配置表格
+			editor_multi.CommonEventTemplateEditCommentJs = function (mod) {
+				editor_multi.lintAutocomplete = true;
+				editor_multi.setLint();
+				editor_multi.importFile(TableFileName);
+			};
 
-        //定義表格操作行為
-        editor_mode.OriginDoActionList = editor_mode.doActionList;
-        editor_mode.doActionList = function (mode, actionList, callback) {
-          if (editor_mode.mode == "CommonEventTemplate") {
-            if (actionList.length == 0) return;
-            printf("修改中...");
-            var cb = function (objs_) {
-              if (objs_.slice(-1)[0] != null) {
-                printe(objs_.slice(-1)[0]);
-                throw objs_.slice(-1)[0];
-              }
-              var str = "修改成功!";
-              if (
-                data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.name ==
-                "template"
-              )
-                str += "<br/>请注意:全塔属性的name尚未修改,请及时予以设置。";
-              printf(str);
-              if (callback) callback();
-            };
-            editor.file.editCommonEventTemplate(actionList, cb);
-          } else {
-            editor_mode.OriginDoActionList(mode, actionList, callback);
-          }
-        };
-        //添加表格列
-        editor.table.CommonEventTemplateAddFunc = function () {
-          let obj = events_c12a15a8_c380_4b28_8144_256cba95f760;
+			//定義表格操作行為
+			editor_mode.OriginDoActionList = editor_mode.doActionList;
+			editor_mode.doActionList = function (mode, actionList, callback) {
+				if (editor_mode.mode == "CommonEventTemplate") {
+					if (actionList.length == 0) return;
+					printf("修改中...");
+					var cb = function (objs_) {
+						if (objs_.slice(-1)[0] != null) {
+							printe(objs_.slice(-1)[0]);
+							throw objs_.slice(-1)[0];
+						}
+						var str = "修改成功!";
+						if (
+							data_a1e2fb4a_e986_4524_b0da_9b7ba7c0874d.firstData.name ==
+							"template"
+						)
+							str += "<br/>请注意:全塔属性的name尚未修改,请及时予以设置。";
+						printf(str);
+						if (callback) callback();
+					};
+					editor.file.editCommonEventTemplate(actionList, cb);
+				} else {
+					editor_mode.OriginDoActionList(mode, actionList, callback);
+				}
+			};
+			//添加表格列
+			editor.table.CommonEventTemplateAddFunc = function () {
+				let obj = events_c12a15a8_c380_4b28_8144_256cba95f760;
 
-          // 1.输入id
-          let newid = prompt("请输入新项的ID(支持中文)");
-          if (newid == null || newid.length == 0) {
-            return;
-          }
+				// 1.输入id
+				let newid = prompt("请输入新项的ID(支持中文)");
+				if (newid == null || newid.length == 0) {
+					return;
+				}
 
-          // 2.检查id是否符合规范或与已有id重复
-          var conflict = true;
-          var basefield = "".replace(/\[[^\[]*\]$/, "");
+				// 2.检查id是否符合规范或与已有id重复
+				var conflict = true;
+				var basefield = "".replace(/\[[^\[]*\]$/, "");
 
-          try {
-            var baseobj = eval("obj" + basefield);
-            conflict = newid in baseobj;
-          } catch (ee) {
-            // 理论上这里不会发生错误
-            printe(ee);
-            throw ee;
-          }
+				try {
+					var baseobj = eval("obj" + basefield);
+					conflict = newid in baseobj;
+				} catch (ee) {
+					// 理论上这里不会发生错误
+					printe(ee);
+					throw ee;
+				}
 
-          if (conflict) {
-            printe("id已存在, 请直接修改该项的值");
-            return;
-          }
+				if (conflict) {
+					printe("id已存在, 请直接修改该项的值");
+					return;
+				}
 
-          // 3.添加
-          editor_mode.addAction(["add", basefield + "['" + newid + "']", null]);
-          editor_mode.onmode("save", function () {
-            printf("添加成功,刷新后生效;也可以继续新增其他项目。");
-          }); //自动保存 删掉此行的话点保存按钮才会保存
-        };
-        //對表格的存讀
-        editor.file.editCommonEventTemplate = function (actionList, callback) {
-          /*actionList:[
+				// 3.添加
+				editor_mode.addAction(["add", basefield + "['" + newid + "']", null]);
+				editor_mode.onmode("save", function () {
+					printf("添加成功,刷新后生效;也可以继续新增其他项目。");
+				}); //自动保存 删掉此行的话点保存按钮才会保存
+			};
+			//對表格的存讀
+			editor.file.editCommonEventTemplate = function (actionList, callback) {
+				/*actionList:[
 				  ["change","['test']",['123']],
 				]
 				为[]时只查询不修改
 				*/
-          var data_obj =
-            events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate;
-          checkCallback(callback);
-          if (isset(actionList) && actionList.length > 0) {
-            actionList.forEach(function (value) {
-              value[1] = "['CommonEventTemplate']" + value[1];
-            });
-            editor.file.saveSetting("events", actionList, function (err) {
-              callback([err]);
-            });
-          } else {
-            callback([
-              Object.assign({}, data_obj),
-              editor.file.CommonEventTemplate_comment._data.CommonEventTemplate,
-              null,
-            ]);
-          }
-        };
-      })();
+				var data_obj =
+					events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate;
+				checkCallback(callback);
+				if (isset(actionList) && actionList.length > 0) {
+					actionList.forEach(function (value) {
+						value[1] = "['CommonEventTemplate']" + value[1];
+					});
+					editor.file.saveSetting("events", actionList, function (err) {
+						callback([err]);
+					});
+				} else {
+					callback([
+						Object.assign({}, data_obj),
+						editor.file.CommonEventTemplate_comment._data.CommonEventTemplate,
+						null,
+					]);
+				}
+			};
+		})();
 
-      //複寫事件編輯器(editor_blocklyconfig)
-      editor_blocklyconfig = function () {
-        // start mark sfergsvae
+		//複寫事件編輯器(editor_blocklyconfig)
+		editor_blocklyconfig = function () {
+				// start mark sfergsvae
 
-        (function () {
-          var getCategory = function (name, custom) {
-            for (var node of document.getElementById("toolbox").children) {
-              if (node.getAttribute("name") == name) return node;
-            }
-            var node = document.createElement("category");
-            node.setAttribute("name", name);
-            if (custom) node.setAttribute("custom", custom);
-            document.getElementById("toolbox").appendChild(node);
-            return node;
-          };
+				(function () {
+					var getCategory = function (name, custom) {
+						for (var node of document.getElementById("toolbox").children) {
+							if (node.getAttribute("name") == name) return node;
+						}
+						var node = document.createElement("category");
+						node.setAttribute("name", name);
+						if (custom) node.setAttribute("custom", custom);
+						document.getElementById("toolbox").appendChild(node);
+						return node;
+					};
 
-          var toolboxObj = {
-            入口方块: [
-              '<label text="入口方块会根据当前类型在此数组中筛选,具体控制在editor_blockly.entranceCategoryCallback中"></label>',
-              MotaActionFunctions.actionParser.parse(
-                [
-                  "欢迎使用事件编辑器",
-                  "本事件触发一次后会消失",
-                  { type: "hide", time: 500 },
-                ],
-                "event"
-              ),
-              MotaActionFunctions.actionParser.parse(
-                {
-                  condition: "flag:__door__===2",
-                  currentFloor: true,
-                  priority: 0,
-                  delayExecute: false,
-                  multiExecute: false,
-                  data: [{ type: "openDoor", loc: [10, 5] }],
-                },
-                "autoEvent"
-              ),
-              MotaActionBlocks["changeFloor_m"].xmlText(),
-              MotaActionFunctions.actionParser.parse(
-                [
-                  {
-                    id: "shop1",
-                    text: "\t[贪婪之神,moneyShop]勇敢的武士啊, 给我${20+2*flag:shop1}金币就可以:",
-                    textInList: "1F金币商店",
-                    choices: [
-                      {
-                        text: "生命+800",
-                        need: "status:money>=20+2*flag:shop1",
-                        action: [
-                          {
-                            type: "comment",
-                            text: "新版商店中需要手动扣减金币和增加访问次数",
-                          },
-                          {
-                            type: "setValue",
-                            name: "status:money",
-                            operator: "-=",
-                            value: "20+2*flag:shop1",
-                          },
-                          {
-                            type: "setValue",
-                            name: "flag:shop1",
-                            operator: "+=",
-                            value: "1",
-                          },
-                          {
-                            type: "setValue",
-                            name: "status:hp",
-                            operator: "+=",
-                            value: "800",
-                          },
-                        ],
-                      },
-                    ],
-                  },
-                  {
-                    id: "itemShop",
-                    item: true,
-                    textInList: "道具商店",
-                    choices: [{ id: "yellowKey", number: 10, money: 10 }],
-                  },
-                  {
-                    id: "keyShop1",
-                    textInList: "回收钥匙商店",
-                    commonEvent: "回收钥匙商店",
-                    args: "",
-                  },
-                ],
-                "shop"
-              ),
-              MotaActionBlocks["common_m"].xmlText(),
-              MotaActionBlocks["beforeBattle_m"].xmlText(),
-              MotaActionBlocks["afterBattle_m"].xmlText(),
-              MotaActionBlocks["afterGetItem_m"].xmlText(),
-              MotaActionBlocks["afterOpenDoor_m"].xmlText(),
-              MotaActionBlocks["firstArrive_m"].xmlText(),
-              MotaActionBlocks["eachArrive_m"].xmlText(),
-              MotaActionBlocks["level_m"].xmlText(),
-              MotaActionFunctions.actionParser.parse(
-                [["MTx", ""]],
-                "floorPartition"
-              ),
-              MotaActionBlocks["commonEvent_m"].xmlText(),
-              MotaActionBlocks["item_m"].xmlText(),
-              MotaActionFunctions.actionParser.parse(
-                [
-                  {
-                    title: "简单",
-                    name: "Easy",
-                    hard: 1,
-                    action: [
-                      { type: "comment", text: "在这里写该难度需执行的事件" },
-                    ],
-                  },
-                ],
-                "levelChoose"
-              ),
-              MotaActionFunctions.actionParser.parse(
-                {
-                  type: 0,
-                  value: { atk: 10 },
-                  percentage: { speed: 10 },
-                },
-                "equip"
-              ),
-              MotaActionFunctions.actionParser.parse(
-                [
-                  {
-                    name: "bg.webp",
-                    x: 0,
-                    y: 0,
-                    canvas: "bg",
-                  },
-                ],
-                "floorImage"
-              ),
-              MotaActionFunctions.actionParser.parse(
-                {
-                  time: 160,
-                  openSound: "door.opus",
-                  closeSound: "door.opus",
-                  keys: { yellowKey: 1, orangeKey: 1 },
-                },
-                "doorInfo"
-              ),
-              MotaActionBlocks["faceIds_m"].xmlText(),
-              MotaActionBlocks["mainStyle_m"].xmlText(),
-              MotaActionFunctions.actionParser.parse(
-                {
-                  背景音乐: "bgm.opus",
-                  确定: "confirm.opus",
-                  攻击: "attack.opus",
-                  背景图: "bg.webp",
-                  领域: "zone",
-                  文件名: "file.jpg",
-                },
-                "nameMap"
-              ),
-              MotaActionFunctions.actionParser.parse(
-                [{ name: "hero.webp", width: 32, height: 32, prefix: "hero_" }],
-                "splitImages"
-              ),
-            ],
-            显示文字: [
-              MotaActionBlocks["text_0_s"].xmlText(),
-              MotaActionBlocks["text_1_s"].xmlText(),
-              MotaActionFunctions.actionParser.parseList(
-                "\t[小妖精,fairy]\f[fairy.webp,0,0]欢迎使用事件编辑器(双击方块可直接预览)"
-              ),
-              MotaActionBlocks["over_s"].xmlText(),
-              MotaActionFunctions.actionParser.parseList([
-                {
-                  type: "overlist",
-                  image: "bg_5043.webp",
-                  memory: false,
-                  hidetime: 30,
-                  list: [
-                    {
-                      text: "",
-                      sound: "",
-                      time: 50,
-                      textColor: "255,255,255,1",
-                      boldColor: "0,0,0,1",
-                      font: "bold 48px Verdana",
-                      frame: 0,
-                    },
-                  ],
-                },
-              ]),
-              MotaActionFunctions.actionParser.parseList([
-                {
-                  type: "cgtext",
-                  bg: "bg_5043.webp",
-                  memory: false,
-                  WindowSkin: false,
-                  head: { name: "face_050445.webp", px: -300 },
-                  name: "菲奥奈",
-                  time: 0,
-                  wait: 2000,
-                  sound: "",
-                  text: "这句话显示在对话框内",
-                  bodyList: [
-                    { name: "tati_050145a.webp", px: 100, filter: false },
-                  ],
-                },
-              ]),
-              MotaActionBlocks["moveTextBox_s"].xmlText(),
-              MotaActionBlocks["clearTextBox_s"].xmlText(),
-              MotaActionBlocks["comment_s"].xmlText(),
-              MotaActionBlocks["autoText_s"].xmlText(),
-              MotaActionBlocks["scrollText_s"].xmlText(),
-              MotaActionBlocks["setText_s"].xmlText(),
-              MotaActionBlocks["tip_s"].xmlText(),
-              MotaActionBlocks["addPop_s"].xmlText(),
-              MotaActionBlocks["confirm_s"].xmlText(),
-              MotaActionBlocks["choices_s"].xmlText([
-                "选择剑或者盾",
-                "流浪者",
-                "man",
-                0,
-                "",
-                MotaActionBlocks["choicesContext"].xmlText([
-                  "剑",
-                  "",
-                  "",
-                  null,
-                  "",
-                  "",
-                  MotaActionFunctions.actionParser.parseList([
-                    { type: "openDoor", loc: [3, 3] },
-                  ]),
-                ]),
-              ]),
-              MotaActionBlocks["win_s"].xmlText(),
-              MotaActionBlocks["lose_s"].xmlText(),
-              MotaActionBlocks["restart_s"].xmlText(),
-            ],
-            数据相关: [
-              MotaActionBlocks["setValue_s"].xmlText([
-                MotaActionBlocks["idIdList_e"].xmlText(["status", "生命"]),
-                "=",
-                "",
-                false,
-              ]),
-              MotaActionBlocks["setEnemy_s"].xmlText(),
-              MotaActionBlocks["setEnemyOnPoint_s"].xmlText(),
-              MotaActionBlocks["resetEnemyOnPoint_s"].xmlText(),
-              MotaActionBlocks["moveEnemyOnPoint_s"].xmlText(),
-              MotaActionBlocks["moveEnemyOnPoint_1_s"].xmlText(),
-              MotaActionBlocks["setEquip_s"].xmlText(),
-              MotaActionBlocks["setFloor_s"].xmlText(),
-              MotaActionBlocks["setGlobalAttribute_s"].xmlText(),
-              MotaActionBlocks["setGlobalValue_s"].xmlText(),
-              MotaActionBlocks["setGlobalFlag_s"].xmlText(),
-              MotaActionBlocks["setNameMap_s"].xmlText(),
-              MotaActionBlocks["input_s"].xmlText(),
-              MotaActionBlocks["input2_s"].xmlText(),
-              MotaActionBlocks["update_s"].xmlText(),
-              MotaActionBlocks["moveAction_s"].xmlText(),
-              MotaActionBlocks["changeFloor_s"].xmlText(),
-              MotaActionBlocks["changePos_s"].xmlText(),
-              MotaActionBlocks["battle_s"].xmlText(),
-              MotaActionBlocks["useItem_s"].xmlText(),
-              MotaActionBlocks["loadEquip_s"].xmlText(),
-              MotaActionBlocks["unloadEquip_s"].xmlText(),
-              MotaActionBlocks["openShop_s"].xmlText(),
-              MotaActionBlocks["disableShop_s"].xmlText(),
-              MotaActionBlocks["setHeroIcon_s"].xmlText(),
-              MotaActionBlocks["follow_s"].xmlText(),
-              MotaActionBlocks["unfollow_s"].xmlText(),
-            ],
-            地图处理: [
-              MotaActionBlocks["battle_1_s"].xmlText(),
-              MotaActionBlocks["openDoor_s"].xmlText(),
-              MotaActionBlocks["closeDoor_s"].xmlText(),
-              MotaActionBlocks["show_s"].xmlText(),
-              MotaActionBlocks["hide_s"].xmlText(),
-              MotaActionBlocks["setBlock_s"].xmlText(),
-              MotaActionBlocks["setBlockOpacity_s"].xmlText(),
-              MotaActionBlocks["setBlockFilter_s"].xmlText(),
-              MotaActionBlocks["turnBlock_s"].xmlText(),
-              MotaActionBlocks["moveHero_s"].xmlText(),
-              MotaActionBlocks["move_s"].xmlText(),
-              MotaActionBlocks["jumpHero_s"].xmlText(),
-              MotaActionBlocks["jumpHero_1_s"].xmlText(),
-              MotaActionBlocks["jump_s"].xmlText(),
-              MotaActionBlocks["jump_1_s"].xmlText(),
-              MotaActionBlocks["showBgFgMap_s"].xmlText(),
-              MotaActionBlocks["hideBgFgMap_s"].xmlText(),
-              MotaActionBlocks["setBgFgBlock_s"].xmlText(),
-              MotaActionBlocks["showFloorImg_s"].xmlText(),
-              MotaActionBlocks["hideFloorImg_s"].xmlText(),
-            ],
-            事件控制: [
-              MotaActionBlocks["if_1_s"].xmlText(),
-              MotaActionBlocks["if_s"].xmlText(),
-              MotaActionFunctions.actionParser.parseList({
-                type: "switch",
-                condition: "判别值",
-                caseList: [
-                  {
-                    action: [
-                      { type: "comment", text: "当判别值是值的场合执行此事件" },
-                    ],
-                  },
-                  {
-                    case: "default",
-                    action: [
-                      {
-                        type: "comment",
-                        text: "当没有符合的值的场合执行default事件",
-                      },
-                    ],
-                  },
-                ],
-              }),
-              MotaActionFunctions.actionParser.parseList({
-                type: "for",
-                name: "temp:A",
-                from: "0",
-                to: "12",
-                step: "1",
-                data: [],
-              }),
-              MotaActionFunctions.actionParser.parseList({
-                type: "forEach",
-                name: "temp:A",
-                list: ["status:atk", "status:def"],
-                data: [],
-              }),
-              MotaActionBlocks["while_s"].xmlText(),
-              MotaActionBlocks["dowhile_s"].xmlText(),
-              MotaActionBlocks["break_s"].xmlText(),
-              MotaActionBlocks["continue_s"].xmlText(),
-              MotaActionBlocks["exit_s"].xmlText(),
-              MotaActionBlocks["trigger_s"].xmlText(),
-              MotaActionBlocks["insert_1_s"].xmlText(),
-              MotaActionBlocks["insert_2_s"].xmlText(),
-            ],
-            特效表现: [
-              MotaActionBlocks["sleep_s"].xmlText(),
-              MotaActionBlocks["changebg_s"].xmlText(),
-              MotaActionFunctions.actionParser.parseList({
-                type: "wait",
-                timeout: 0,
-                data: [
-                  {
-                    case: "keyboard",
-                    keycode: "13,32",
-                    action: [
-                      {
-                        type: "comment",
-                        text: "当按下回车(keycode=13)或空格(keycode=32)时执行此事件\n超时剩余时间会写入flag:timeout",
-                      },
-                    ],
-                  },
-                  {
-                    case: "mouse",
-                    px: [0, 32],
-                    py: [0, 32],
-                    action: [
-                      {
-                        type: "comment",
-                        text: "当点击地图左上角时执行此事件\n超时剩余时间会写入flag:timeout",
-                      },
-                    ],
-                  },
-                  {
-                    case: "condition",
-                    condition: "flag:type==0\n&&flag:keycode==13",
-                    action: [
-                      {
-                        type: "comment",
-                        text: "当满足自定义条件时会执行此事件\n超时剩余时间会写入flag:timeout",
-                      },
-                    ],
-                  },
-                  {
-                    case: "timeout",
-                    action: [
-                      { type: "comment", text: "当超时未操作时执行此事件" },
-                    ],
-                  },
-                ],
-              }),
-              MotaActionBlocks["waitAsync_s"].xmlText(),
-              MotaActionBlocks["stopAsync_s"].xmlText(),
-              MotaActionBlocks["op_s"].xmlText(),
-              MotaActionBlocks["drawWarning_s"].xmlText(),
-              MotaActionBlocks["changeMouse_s"].xmlText(),
-              MotaActionBlocks["removeMouse_s"].xmlText(),
-              MotaActionBlocks["vibrate_s"].xmlText(),
-              MotaActionBlocks["setanimate_s"].xmlText(),
-              MotaActionBlocks["deleteanimate_s"].xmlText(),
-              MotaActionBlocks["playanimate_s"].xmlText(),
-              MotaActionBlocks["clearanimate_s"].xmlText(),
-              MotaActionBlocks["animate_s"].xmlText(),
-              MotaActionBlocks["animate_1_s"].xmlText(),
-              MotaActionBlocks["stopAnimate_s"].xmlText(),
-              MotaActionBlocks["setViewport_s"].xmlText(),
-              MotaActionBlocks["setViewport_1_s"].xmlText(),
-              MotaActionBlocks["lockViewport_s"].xmlText(),
-              MotaActionBlocks["showStatusBar_s"].xmlText(),
-              MotaActionBlocks["hideStatusBar_s"].xmlText(),
-              MotaActionBlocks["setHeroOpacity_s"].xmlText(),
-              MotaActionBlocks["setCurtain_0_s"].xmlText(),
-              MotaActionBlocks["setCurtain_1_s"].xmlText(),
-              MotaActionBlocks["screenFlash_s"].xmlText(),
-              MotaActionBlocks["setWeather_s"].xmlText(),
-              MotaActionBlocks["callBook_s"].xmlText(),
-              MotaActionBlocks["callSave_s"].xmlText(),
-              MotaActionBlocks["autoSave_s"].xmlText(),
-              MotaActionBlocks["forbidSave_s"].xmlText(),
-              MotaActionBlocks["callLoad_s"].xmlText(),
-            ],
-            音像处理: [
-              MotaActionBlocks["animationDrawable_s"].xmlText(),
-              MotaActionBlocks["setanimate_s"].xmlText(),
-              MotaActionBlocks["deleteanimate_s"].xmlText(),
-              MotaActionBlocks["playanimate_s"].xmlText(),
-              MotaActionBlocks["clearanimate_s"].xmlText(),
-              MotaActionBlocks["showImage_s"].xmlText(),
-              MotaActionBlocks["showImage_1_s"].xmlText(),
-              MotaActionBlocks["hideImage_s"].xmlText(),
-              MotaActionBlocks["showTextImage_s"].xmlText(),
-              MotaActionBlocks["moveImage_s"].xmlText(),
-              MotaActionBlocks["rotateImage_s"].xmlText(),
-              MotaActionBlocks["scaleImage_s"].xmlText(),
-              MotaActionBlocks["showGif_s"].xmlText(),
-              MotaActionBlocks["playBgm_s"].xmlText(),
-              MotaActionBlocks["playStereo_s"].xmlText(),
-              MotaActionBlocks["moveStereo_s"].xmlText(),
-              MotaActionBlocks["pauseBgm_s"].xmlText(),
-              MotaActionBlocks["resumeBgm_s"].xmlText(),
-              MotaActionBlocks["loadBgm_s"].xmlText(),
-              MotaActionBlocks["freeBgm_s"].xmlText(),
-              MotaActionBlocks["playSound_s"].xmlText(),
-              MotaActionBlocks["playSound_1_s"].xmlText(),
-              MotaActionBlocks["stopSound_s"].xmlText(),
-              MotaActionBlocks["setVolume_s"].xmlText(),
-              MotaActionBlocks["setBgmSpeed_s"].xmlText(),
-            ],
-            UI绘制: [
-              MotaActionBlocks["previewUI_s"].xmlText(),
-              MotaActionBlocks["clearMap_s"].xmlText(),
-              MotaActionBlocks["setAttribute_s"].xmlText(),
-              MotaActionBlocks["setFilter_s"].xmlText(),
-              MotaActionBlocks["fillText_s"].xmlText(),
-              MotaActionBlocks["fillBoldText_s"].xmlText(),
-              MotaActionBlocks["drawTextContent_s"].xmlText(),
-              MotaActionBlocks["fillRect_s"].xmlText(),
-              MotaActionBlocks["strokeRect_s"].xmlText(),
-              MotaActionBlocks["drawLine_s"].xmlText(),
-              MotaActionBlocks["drawArrow_s"].xmlText(),
-              MotaActionBlocks["fillPolygon_s"].xmlText(),
-              MotaActionBlocks["strokePolygon_s"].xmlText(),
-              MotaActionBlocks["fillEllipse_s"].xmlText(),
-              MotaActionBlocks["strokeEllipse_s"].xmlText(),
-              MotaActionBlocks["fillArc_s"].xmlText(),
-              MotaActionBlocks["strokeArc_s"].xmlText(),
-              MotaActionBlocks["drawImage_s"].xmlText(),
-              MotaActionBlocks["drawImage_1_s"].xmlText(),
-              MotaActionBlocks["drawIcon_s"].xmlText(),
-              MotaActionBlocks["drawBackground_s"].xmlText(),
-              MotaActionBlocks["drawSelector_s"].xmlText(),
-              MotaActionBlocks["drawSelector_1_s"].xmlText(),
-            ],
-            原生脚本: [
-              MotaActionBlocks["function_s"].xmlText(),
-              MotaActionBlocks["unknown_s"].xmlText(),
-            ],
-            值块: [
-              MotaActionBlocks["setValue_s"].xmlText([
-                MotaActionBlocks["idIdList_e"].xmlText(["status", "生命"]),
-                "=",
-                "",
-                false,
-              ]),
-              MotaActionBlocks["expression_arithmetic_0"].xmlText(),
-              MotaActionBlocks["idFlag_e"].xmlText(),
-              MotaActionBlocks["idTemp_e"].xmlText(),
-              MotaActionBlocks["negate_e"].xmlText(),
-              MotaActionBlocks["unaryOperation_e"].xmlText(),
-              MotaActionBlocks["bool_e"].xmlText(),
-              MotaActionBlocks["idString_e"].xmlText(),
-              MotaActionBlocks["idIdList_e"].xmlText(),
-              MotaActionBlocks["idFixedList_e"].xmlText(),
-              MotaActionBlocks["enemyattr_e"].xmlText(),
-              MotaActionBlocks["blockId_e"].xmlText(),
-              MotaActionBlocks["blockNumber_e"].xmlText(),
-              MotaActionBlocks["blockCls_e"].xmlText(),
-              MotaActionBlocks["hasEquip_e"].xmlText(),
-              MotaActionBlocks["equip_e"].xmlText(),
-              MotaActionBlocks["nextXY_e"].xmlText(),
-              MotaActionBlocks["isReplaying_e"].xmlText(),
-              MotaActionBlocks["hasVisitedFloor_e"].xmlText(),
-              MotaActionBlocks["isShopVisited_e"].xmlText(),
-              MotaActionBlocks["canBattle_e"].xmlText(),
-              MotaActionBlocks["damage_e"].xmlText(),
-              MotaActionBlocks["damage_1_e"].xmlText(),
-              MotaActionBlocks["rand_e"].xmlText(),
-              MotaActionBlocks["evalString_e"].xmlText(),
-            ],
-            常见事件模板: [
-              '<label text="此处只是占位符,实际定义在#region 動態常見事件模板"></label>',
-            ],
-            最近使用事件: [
-              '<label text="此处只是占位符,实际定义在editor_blockly.searchBlockCategoryCallback中"></label>',
-            ],
-          };
-          var toolboxgap = '<sep gap="5"></sep>';
-          //xml_text = MotaActionFunctions.actionParser.parse(obj,type||'event')
-          //MotaActionBlocks['idString_e'].xmlText()
+					var toolboxObj = {
+						入口方块: [
+							'<label text="入口方块会根据当前类型在此数组中筛选,具体控制在editor_blockly.entranceCategoryCallback中"></label>',
+							MotaActionFunctions.actionParser.parse(
+								[
+									"欢迎使用事件编辑器",
+									"本事件触发一次后会消失",
+									{ type: "hide", time: 500 },
+								],
+								"event"
+							),
+							MotaActionFunctions.actionParser.parse({
+									condition: "flag:__door__===2",
+									currentFloor: true,
+									priority: 0,
+									delayExecute: false,
+									multiExecute: false,
+									data: [{ type: "openDoor", loc: [10, 5] }],
+								},
+								"autoEvent"
+							),
+							MotaActionBlocks["changeFloor_m"].xmlText(),
+							MotaActionFunctions.actionParser.parse(
+								[{
+										id: "shop1",
+										text: "\t[贪婪之神,moneyShop]勇敢的武士啊, 给我${20+2*flag:shop1}金币就可以:",
+										textInList: "1F金币商店",
+										choices: [{
+											text: "生命+800",
+											need: "status:money>=20+2*flag:shop1",
+											action: [{
+													type: "comment",
+													text: "新版商店中需要手动扣减金币和增加访问次数",
+												},
+												{
+													type: "setValue",
+													name: "status:money",
+													operator: "-=",
+													value: "20+2*flag:shop1",
+												},
+												{
+													type: "setValue",
+													name: "flag:shop1",
+													operator: "+=",
+													value: "1",
+												},
+												{
+													type: "setValue",
+													name: "status:hp",
+													operator: "+=",
+													value: "800",
+												},
+											],
+										}, ],
+									},
+									{
+										id: "itemShop",
+										item: true,
+										textInList: "道具商店",
+										choices: [{ id: "yellowKey", number: 10, money: 10 }],
+									},
+									{
+										id: "keyShop1",
+										textInList: "回收钥匙商店",
+										commonEvent: "回收钥匙商店",
+										args: "",
+									},
+								],
+								"shop"
+							),
+							MotaActionBlocks["common_m"].xmlText(),
+							MotaActionBlocks["beforeBattle_m"].xmlText(),
+							MotaActionBlocks["afterBattle_m"].xmlText(),
+							MotaActionBlocks["afterGetItem_m"].xmlText(),
+							MotaActionBlocks["afterOpenDoor_m"].xmlText(),
+							MotaActionBlocks["firstArrive_m"].xmlText(),
+							MotaActionBlocks["eachArrive_m"].xmlText(),
+							MotaActionBlocks["level_m"].xmlText(),
+							MotaActionFunctions.actionParser.parse(
+								[
+									["MTx", ""]
+								],
+								"floorPartition"
+							),
+							MotaActionBlocks["commonEvent_m"].xmlText(),
+							MotaActionBlocks["item_m"].xmlText(),
+							MotaActionFunctions.actionParser.parse(
+								[{
+									title: "简单",
+									name: "Easy",
+									hard: 1,
+									action: [
+										{ type: "comment", text: "在这里写该难度需执行的事件" },
+									],
+								}, ],
+								"levelChoose"
+							),
+							MotaActionFunctions.actionParser.parse({
+									type: 0,
+									value: { atk: 10 },
+									percentage: { speed: 10 },
+								},
+								"equip"
+							),
+							MotaActionFunctions.actionParser.parse(
+								[{
+									name: "bg.webp",
+									x: 0,
+									y: 0,
+									canvas: "bg",
+								}, ],
+								"floorImage"
+							),
+							MotaActionFunctions.actionParser.parse({
+									time: 160,
+									openSound: "door.opus",
+									closeSound: "door.opus",
+									keys: { yellowKey: 1, orangeKey: 1 },
+								},
+								"doorInfo"
+							),
+							MotaActionBlocks["faceIds_m"].xmlText(),
+							MotaActionBlocks["mainStyle_m"].xmlText(),
+							MotaActionFunctions.actionParser.parse({
+									背景音乐: "bgm.opus",
+									确定: "confirm.opus",
+									攻击: "attack.opus",
+									背景图: "bg.webp",
+									领域: "zone",
+									文件名: "file.jpg",
+								},
+								"nameMap"
+							),
+							MotaActionFunctions.actionParser.parse(
+								[{ name: "hero.webp", width: 32, height: 32, prefix: "hero_" }],
+								"splitImages"
+							),
+						],
+						显示文字: [
+							MotaActionBlocks["text_0_s"].xmlText(),
+							MotaActionBlocks["text_1_s"].xmlText(),
+							MotaActionFunctions.actionParser.parseList(
+								"\t[小妖精,fairy]\f[fairy.webp,0,0]欢迎使用事件编辑器(双击方块可直接预览)"
+							),
+							MotaActionBlocks["over_s"].xmlText(),
+							MotaActionFunctions.actionParser.parseList([{
+								type: "overlist",
+								image: "bg_5043.webp",
+								memory: false,
+								hidetime: 30,
+								list: [{
+									text: "",
+									sound: "",
+									time: 50,
+									textColor: "255,255,255,1",
+									boldColor: "0,0,0,1",
+									font: "bold 48px Verdana",
+									frame: 0,
+								}, ],
+							}, ]),
+							MotaActionBlocks["cgtextList_s"].xmlText(),
+							MotaActionFunctions.actionParser.parseList([{
+								type: "cgtext",
+								bg: "bg_5043.webp",
+								memory: false,
+								WindowSkin: false,
+								index: 0,
+								head: { name: "face_050445.webp", px: -300 },
 
-          //#region 動態常見事件模板
-          let CommonEventTemplateHTML = [];
+								time: 0,
+								wait: 2000,
+								sound: "",
 
-          for (let commonEventName in events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate) {
-            if (
-              events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate.hasOwnProperty(
-                commonEventName
-              )
-            ) {
-              let actionParserJson = Array.from(
-                events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate[
-                  commonEventName
-                ] ?? []
-              );
+								bodyList: [
+									{ name: "tati_050145a.webp", px: 100, filter: false },
+								],
+							}, ]),
+							MotaActionBlocks["moveTextBox_s"].xmlText(),
+							MotaActionBlocks["clearTextBox_s"].xmlText(),
+							MotaActionBlocks["comment_s"].xmlText(),
+							MotaActionBlocks["autoText_s"].xmlText(),
+							MotaActionBlocks["scrollText_s"].xmlText(),
+							MotaActionBlocks["setText_s"].xmlText(),
+							MotaActionBlocks["tip_s"].xmlText(),
+							MotaActionBlocks["addPop_s"].xmlText(),
+							MotaActionBlocks["confirm_s"].xmlText(),
+							MotaActionBlocks["choices_s"].xmlText([
+								"选择剑或者盾",
+								"流浪者",
+								"man",
+								0,
+								"",
+								MotaActionBlocks["choicesContext"].xmlText([
+									"剑",
+									"",
+									"",
+									null,
+									"",
+									"",
+									MotaActionFunctions.actionParser.parseList([
+										{ type: "openDoor", loc: [3, 3] },
+									]),
+								]),
+							]),
+							MotaActionBlocks["win_s"].xmlText(),
+							MotaActionBlocks["lose_s"].xmlText(),
+							MotaActionBlocks["restart_s"].xmlText(),
+						],
+						数据相关: [
+							MotaActionBlocks["setValue_s"].xmlText([
+								MotaActionBlocks["idIdList_e"].xmlText(["status", "生命"]),
+								"=",
+								"",
+								false,
+							]),
+							MotaActionBlocks["setEnemy_s"].xmlText(),
+							MotaActionBlocks["setEnemyOnPoint_s"].xmlText(),
+							MotaActionBlocks["resetEnemyOnPoint_s"].xmlText(),
+							MotaActionBlocks["moveEnemyOnPoint_s"].xmlText(),
+							MotaActionBlocks["moveEnemyOnPoint_1_s"].xmlText(),
+							MotaActionBlocks["setEquip_s"].xmlText(),
+							MotaActionBlocks["setFloor_s"].xmlText(),
+							MotaActionBlocks["setGlobalAttribute_s"].xmlText(),
+							MotaActionBlocks["setGlobalValue_s"].xmlText(),
+							MotaActionBlocks["setGlobalFlag_s"].xmlText(),
+							MotaActionBlocks["setNameMap_s"].xmlText(),
+							MotaActionBlocks["input_s"].xmlText(),
+							MotaActionBlocks["input2_s"].xmlText(),
+							MotaActionBlocks["update_s"].xmlText(),
+							MotaActionBlocks["moveAction_s"].xmlText(),
+							MotaActionBlocks["changeFloor_s"].xmlText(),
+							MotaActionBlocks["changePos_s"].xmlText(),
+							MotaActionBlocks["battle_s"].xmlText(),
+							MotaActionBlocks["useItem_s"].xmlText(),
+							MotaActionBlocks["loadEquip_s"].xmlText(),
+							MotaActionBlocks["unloadEquip_s"].xmlText(),
+							MotaActionBlocks["openShop_s"].xmlText(),
+							MotaActionBlocks["disableShop_s"].xmlText(),
+							MotaActionBlocks["setHeroIcon_s"].xmlText(),
+							MotaActionBlocks["follow_s"].xmlText(),
+							MotaActionBlocks["unfollow_s"].xmlText(),
+						],
+						地图处理: [
+							MotaActionBlocks["battle_1_s"].xmlText(),
+							MotaActionBlocks["openDoor_s"].xmlText(),
+							MotaActionBlocks["closeDoor_s"].xmlText(),
+							MotaActionBlocks["show_s"].xmlText(),
+							MotaActionBlocks["hide_s"].xmlText(),
+							MotaActionBlocks["setBlock_s"].xmlText(),
+							MotaActionBlocks["setBlockOpacity_s"].xmlText(),
+							MotaActionBlocks["setBlockFilter_s"].xmlText(),
+							MotaActionBlocks["turnBlock_s"].xmlText(),
+							MotaActionBlocks["moveHero_s"].xmlText(),
+							MotaActionBlocks["move_s"].xmlText(),
+							MotaActionBlocks["jumpHero_s"].xmlText(),
+							MotaActionBlocks["jumpHero_1_s"].xmlText(),
+							MotaActionBlocks["jump_s"].xmlText(),
+							MotaActionBlocks["jump_1_s"].xmlText(),
+							MotaActionBlocks["showBgFgMap_s"].xmlText(),
+							MotaActionBlocks["hideBgFgMap_s"].xmlText(),
+							MotaActionBlocks["setBgFgBlock_s"].xmlText(),
+							MotaActionBlocks["showFloorImg_s"].xmlText(),
+							MotaActionBlocks["hideFloorImg_s"].xmlText(),
+						],
+						事件控制: [
+							MotaActionBlocks["if_1_s"].xmlText(),
+							MotaActionBlocks["if_s"].xmlText(),
+							MotaActionFunctions.actionParser.parseList({
+								type: "switch",
+								condition: "判别值",
+								caseList: [{
+										action: [
+											{ type: "comment", text: "当判别值是值的场合执行此事件" },
+										],
+									},
+									{
+										case: "default",
+										action: [{
+											type: "comment",
+											text: "当没有符合的值的场合执行default事件",
+										}, ],
+									},
+								],
+							}),
+							MotaActionFunctions.actionParser.parseList({
+								type: "for",
+								name: "temp:A",
+								from: "0",
+								to: "12",
+								step: "1",
+								data: [],
+							}),
+							MotaActionFunctions.actionParser.parseList({
+								type: "forEach",
+								name: "temp:A",
+								list: ["status:atk", "status:def"],
+								data: [],
+							}),
+							MotaActionBlocks["while_s"].xmlText(),
+							MotaActionBlocks["dowhile_s"].xmlText(),
+							MotaActionBlocks["break_s"].xmlText(),
+							MotaActionBlocks["continue_s"].xmlText(),
+							MotaActionBlocks["exit_s"].xmlText(),
+							MotaActionBlocks["trigger_s"].xmlText(),
+							MotaActionBlocks["insert_1_s"].xmlText(),
+							MotaActionBlocks["insert_2_s"].xmlText(),
+						],
+						特效表现: [
+							MotaActionBlocks["sleep_s"].xmlText(),
+							MotaActionBlocks["setq_s"].xmlText(),
+							MotaActionBlocks["setcgs_s"].xmlText(),
+							MotaActionBlocks["setmusics_s"].xmlText(),
+							MotaActionBlocks["changebg_s"].xmlText(),
+							MotaActionBlocks["drawbackground_s"].xmlText(),
+							MotaActionBlocks["clearbackground_s"].xmlText(),
+							MotaActionFunctions.actionParser.parseList({
+								type: "wait",
+								timeout: 0,
+								data: [{
+										case: "keyboard",
+										keycode: "13,32",
+										action: [{
+											type: "comment",
+											text: "当按下回车(keycode=13)或空格(keycode=32)时执行此事件\n超时剩余时间会写入flag:timeout",
+										}, ],
+									},
+									{
+										case: "mouse",
+										px: [0, 32],
+										py: [0, 32],
+										action: [{
+											type: "comment",
+											text: "当点击地图左上角时执行此事件\n超时剩余时间会写入flag:timeout",
+										}, ],
+									},
+									{
+										case: "condition",
+										condition: "flag:type==0\n&&flag:keycode==13",
+										action: [{
+											type: "comment",
+											text: "当满足自定义条件时会执行此事件\n超时剩余时间会写入flag:timeout",
+										}, ],
+									},
+									{
+										case: "timeout",
+										action: [
+											{ type: "comment", text: "当超时未操作时执行此事件" },
+										],
+									},
+								],
+							}),
+							MotaActionBlocks["waitAsync_s"].xmlText(),
+							MotaActionBlocks["stopAsync_s"].xmlText(),
+							MotaActionBlocks["op_s"].xmlText(),
+							MotaActionBlocks["drawWarning_s"].xmlText(),
+							MotaActionBlocks["changeMouse_s"].xmlText(),
+							MotaActionBlocks["removeMouse_s"].xmlText(),
+							MotaActionBlocks["vibrate_s"].xmlText(),
+							MotaActionBlocks["animate_s"].xmlText(),
+							MotaActionBlocks["animate_1_s"].xmlText(),
+							MotaActionBlocks["animateResize_s"].xmlText(),
+							MotaActionBlocks["moveAnimate_s"].xmlText(),
+							MotaActionBlocks["pauseAnimate_s"].xmlText(),
+							MotaActionBlocks["remuseAnimate_s"].xmlText(),
+							MotaActionBlocks["stopAnimate_s"].xmlText(),
+							MotaActionBlocks["setViewport_s"].xmlText(),
+							MotaActionBlocks["setViewport_1_s"].xmlText(),
+							MotaActionBlocks["lockViewport_s"].xmlText(),
+							MotaActionBlocks["showStatusBar_s"].xmlText(),
+							MotaActionBlocks["hideStatusBar_s"].xmlText(),
+							MotaActionBlocks["setHeroOpacity_s"].xmlText(),
+							MotaActionBlocks["setCurtain_0_s"].xmlText(),
+							MotaActionBlocks["setCurtain_1_s"].xmlText(),
+							MotaActionBlocks["screenFlash_s"].xmlText(),
+							MotaActionBlocks["setWeather_s"].xmlText(),
+							MotaActionBlocks["callBook_s"].xmlText(),
+							MotaActionBlocks["callSave_s"].xmlText(),
+							MotaActionBlocks["autoSave_s"].xmlText(),
+							MotaActionBlocks["forbidSave_s"].xmlText(),
+							MotaActionBlocks["callLoad_s"].xmlText(),
+						],
+						音像处理: [
+							MotaActionBlocks["animationDrawable_s"].xmlText(),
+							MotaActionBlocks["setanimate_s"].xmlText(),
+							MotaActionBlocks["deleteanimate_s"].xmlText(),
+							MotaActionBlocks["playanimate_s"].xmlText(),
+							MotaActionBlocks["clearanimate_s"].xmlText(),
+							MotaActionBlocks["animateloop_s"].xmlText(),
+							MotaActionBlocks["animatereverse_s"].xmlText(),
+							MotaActionBlocks["animatepause_s"].xmlText(),
+							MotaActionBlocks["animatemove_s"].xmlText(),
+							MotaActionBlocks["introAndLoop_s"].xmlText(),
+							MotaActionBlocks["showImage_s"].xmlText(),
+							MotaActionBlocks["showImage_1_s"].xmlText(),
+							MotaActionBlocks["hideImage_s"].xmlText(),
+							MotaActionBlocks["showTextImage_s"].xmlText(),
+							MotaActionBlocks["moveImage_s"].xmlText(),
+							MotaActionBlocks["rotateImage_s"].xmlText(),
+							MotaActionBlocks["scaleImage_s"].xmlText(),
+							MotaActionBlocks["showGif_s"].xmlText(),
+							MotaActionBlocks["playBgm_s"].xmlText(),
+							MotaActionBlocks["pauseBgm_s"].xmlText(),
+							MotaActionBlocks["resumeBgm_s"].xmlText(),
+							MotaActionBlocks["loadBgm_s"].xmlText(),
+							MotaActionBlocks["freeBgm_s"].xmlText(),
+							MotaActionBlocks["playSound_s"].xmlText(),
+							MotaActionBlocks["playSound_1_s"].xmlText(),
+							MotaActionBlocks["stopSound_s"].xmlText(),
+							MotaActionBlocks["setVolume_s"].xmlText(),
+							MotaActionBlocks["setBgmSpeed_s"].xmlText(),
+						],
+						UI绘制: [
+							MotaActionBlocks["previewUI_s"].xmlText(),
+							MotaActionBlocks["clearMap_s"].xmlText(),
+							MotaActionBlocks["setAttribute_s"].xmlText(),
+							MotaActionBlocks["setFilter_s"].xmlText(),
+							MotaActionBlocks["fillText_s"].xmlText(),
+							MotaActionBlocks["fillBoldText_s"].xmlText(),
+							MotaActionBlocks["drawTextContent_s"].xmlText(),
+							MotaActionBlocks["fillRect_s"].xmlText(),
+							MotaActionBlocks["strokeRect_s"].xmlText(),
+							MotaActionBlocks["drawLine_s"].xmlText(),
+							MotaActionBlocks["drawArrow_s"].xmlText(),
+							MotaActionBlocks["fillPolygon_s"].xmlText(),
+							MotaActionBlocks["strokePolygon_s"].xmlText(),
+							MotaActionBlocks["fillEllipse_s"].xmlText(),
+							MotaActionBlocks["strokeEllipse_s"].xmlText(),
+							MotaActionBlocks["fillArc_s"].xmlText(),
+							MotaActionBlocks["strokeArc_s"].xmlText(),
+							MotaActionBlocks["drawImage_s"].xmlText(),
+							MotaActionBlocks["drawImage_1_s"].xmlText(),
+							MotaActionBlocks["drawIcon_s"].xmlText(),
+							MotaActionBlocks["drawBackground_s"].xmlText(),
+							MotaActionBlocks["drawSelector_s"].xmlText(),
+							MotaActionBlocks["drawSelector_1_s"].xmlText(),
+						],
+						原生脚本: [
+							MotaActionBlocks["function_s"].xmlText(),
+							MotaActionBlocks["unknown_s"].xmlText(),
+						],
+						值块: [
+							MotaActionBlocks["setValue_s"].xmlText([
+								MotaActionBlocks["idIdList_e"].xmlText(["status", "生命"]),
+								"=",
+								"",
+								false,
+							]),
+							MotaActionBlocks["expression_arithmetic_0"].xmlText(),
+							MotaActionBlocks["idFlag_e"].xmlText(),
+							MotaActionBlocks["idTemp_e"].xmlText(),
+							MotaActionBlocks["negate_e"].xmlText(),
+							MotaActionBlocks["unaryOperation_e"].xmlText(),
+							MotaActionBlocks["bool_e"].xmlText(),
+							MotaActionBlocks["idString_e"].xmlText(),
+							MotaActionBlocks["idIdList_e"].xmlText(),
+							MotaActionBlocks["idFixedList_e"].xmlText(),
+							MotaActionBlocks["enemyattr_e"].xmlText(),
+							MotaActionBlocks["blockId_e"].xmlText(),
+							MotaActionBlocks["blockNumber_e"].xmlText(),
+							MotaActionBlocks["blockCls_e"].xmlText(),
+							MotaActionBlocks["hasEquip_e"].xmlText(),
+							MotaActionBlocks["equip_e"].xmlText(),
+							MotaActionBlocks["nextXY_e"].xmlText(),
+							MotaActionBlocks["isReplaying_e"].xmlText(),
+							MotaActionBlocks["hasVisitedFloor_e"].xmlText(),
+							MotaActionBlocks["isShopVisited_e"].xmlText(),
+							MotaActionBlocks["canBattle_e"].xmlText(),
+							MotaActionBlocks["damage_e"].xmlText(),
+							MotaActionBlocks["damage_1_e"].xmlText(),
+							MotaActionBlocks["rand_e"].xmlText(),
+							MotaActionBlocks["evalString_e"].xmlText(),
+						],
+						常见事件模板: [
+							'<label text="此处只是占位符,实际定义在#region 動態常見事件模板"></label>',
+						],
+						最近使用事件: [
+							'<label text="此处只是占位符,实际定义在editor_blockly.searchBlockCategoryCallback中"></label>',
+						],
+					};
+					var toolboxgap = '<sep gap="5"></sep>';
+					//xml_text = MotaActionFunctions.actionParser.parse(obj,type||'event')
+					//MotaActionBlocks['idString_e'].xmlText()
 
-              let labelHTML = "";
-              let blockHTML = "";
+					//#region 動態常見事件模板
+					let CommonEventTemplateHTML = [];
 
-              labelHTML = `<label text="${commonEventName}"></label>`;
+					for (let commonEventName in events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate) {
+						if (
+							events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate.hasOwnProperty(
+								commonEventName
+							)
+						) {
+							let actionParserJson = Array.from(
+								events_c12a15a8_c380_4b28_8144_256cba95f760.CommonEventTemplate[
+									commonEventName
+								] ?? []
+							);
 
-              if (actionParserJson.length > 1) {
-                actionParserJson = {
-                  type: "if",
-                  condition: "true",
-                  true: actionParserJson,
-                };
-              } else if (actionParserJson.length < 1) {
-                actionParserJson = [
-                  "空的常用事件模板。\n請在主頁下拉菜單中,選擇常用事件模板,進行編輯。\n編輯後需按F5刷新事件編輯器。",
-                ];
-              }
-              blockHTML =
-                MotaActionFunctions.actionParser.parseList(actionParserJson);
+							let labelHTML = "";
+							let blockHTML = "";
 
-              CommonEventTemplateHTML.push(labelHTML);
-              CommonEventTemplateHTML.push(blockHTML);
-            }
-          }
+							labelHTML = `<label text="${commonEventName}"></label>`;
 
-          toolboxObj["常见事件模板"] = CommonEventTemplateHTML;
-          //#endregion
+							if (actionParserJson.length > 1) {
+								actionParserJson = {
+									type: "if",
+									condition: "true",
+									true: actionParserJson,
+								};
+							} else if (actionParserJson.length < 1) {
+								actionParserJson = [
+									"空的常用事件模板。\n請在主頁下拉菜單中,選擇常用事件模板,進行編輯。\n編輯後需按F5刷新事件編輯器。",
+								];
+							}
+							blockHTML =
+								MotaActionFunctions.actionParser.parseList(actionParserJson);
 
-          for (var name in toolboxObj) {
-            var custom = null;
-            if (name == "最近使用事件") custom = "searchBlockCategory";
-            if (name == "入口方块") custom = "entranceCategory";
-            getCategory(name, custom).innerHTML =
-              toolboxObj[name].join(toolboxgap);
-          }
+							CommonEventTemplateHTML.push(labelHTML);
+							CommonEventTemplateHTML.push(blockHTML);
+						}
+					}
 
-          var blocklyArea = document.getElementById("blocklyArea");
-          var blocklyDiv = document.getElementById("blocklyDiv");
-          var workspace = Blockly.inject(blocklyDiv, {
-            media: "_server/blockly/media/",
-            toolbox: document.getElementById("toolbox"),
-            zoom: {
-              controls: true,
-              wheel: false, //滚轮改为上下(shift:左右)翻滚
-              startScale: 1.0,
-              maxScale: 3,
-              minScale: 0.3,
-              scaleSpeed: 1.08,
-            },
-            trashcan: false,
-          });
+					toolboxObj["常见事件模板"] = CommonEventTemplateHTML;
+					//#endregion
 
-          editor_blockly.isCommonEntry = function () {
-            var commonEntries = [
-              "beforeBattle",
-              "afterBattle",
-              "afterOpenDoor",
-              "firstArrive",
-              "eachArrive",
-              "commonEvent",
-              "item",
-            ];
-            return commonEntries.indexOf(editor_blockly.entryType) >= 0;
-          };
+					for (var name in toolboxObj) {
+						var custom = null;
+						if (name == "最近使用事件") custom = "searchBlockCategory";
+						if (name == "入口方块") custom = "entranceCategory";
+						getCategory(name, custom).innerHTML =
+							toolboxObj[name].join(toolboxgap);
+					}
 
-          editor_blockly.entranceCategoryCallback = function (workspace) {
-            var list = toolboxObj["入口方块"];
-            var xmlList = [];
-            var eventType =
-              (editor_blockly.isCommonEntry()
-                ? "common"
-                : editor_blockly.entryType) + "_m";
-            for (var ii = 0, blockText; (blockText = list[ii]); ii++) {
-              if (
-                new RegExp('<block type="' + eventType + '">').exec(blockText)
-              ) {
-                var block = Blockly.Xml.textToDom(
-                  "<xml>" + blockText + "</xml>"
-                ).firstChild;
-                block.setAttribute("gap", 5);
-                xmlList.push(block);
-              }
-            }
-            return xmlList;
-          };
+					var blocklyArea = document.getElementById("blocklyArea");
+					var blocklyDiv = document.getElementById("blocklyDiv");
+					var workspace = Blockly.inject(blocklyDiv, {
+						media: "_server/blockly/media/",
+						toolbox: document.getElementById("toolbox"),
+						zoom: {
+							controls: true,
+							wheel: false, //滚轮改为上下(shift:左右)翻滚
+							startScale: 1.0,
+							maxScale: 3,
+							minScale: 0.3,
+							scaleSpeed: 1.08,
+						},
+						trashcan: false,
+					});
 
-          workspace.registerToolboxCategoryCallback(
-            "entranceCategory",
-            editor_blockly.entranceCategoryCallback
-          );
+					editor_blockly.isCommonEntry = function () {
+						var commonEntries = [
+							"beforeBattle",
+							"afterBattle",
+							"afterOpenDoor",
+							"firstArrive",
+							"eachArrive",
+							"commonEvent",
+							"item",
+						];
+						return commonEntries.indexOf(editor_blockly.entryType) >= 0;
+					};
 
-          editor_blockly.searchBlockCategoryCallback = function (workspace) {
-            var xmlList = [];
-            var labels = editor_blockly.searchBlock();
-            for (var i = 0; i < labels.length; i++) {
-              var blockText =
-                "<xml>" + MotaActionBlocks[labels[i]].xmlText() + "</xml>";
-              var block = Blockly.Xml.textToDom(blockText).firstChild;
-              block.setAttribute("gap", 5);
-              xmlList.push(block);
-            }
-            return xmlList;
-          };
+					editor_blockly.entranceCategoryCallback = function (workspace) {
+						var list = toolboxObj["入口方块"];
+						var xmlList = [];
+						var eventType =
+							(editor_blockly.isCommonEntry() ?
+								"common" :
+								editor_blockly.entryType) + "_m";
+						for (var ii = 0, blockText;
+							(blockText = list[ii]); ii++) {
+							if (
+								new RegExp('<block type="' + eventType + '">').exec(blockText)
+							) {
+								var block = Blockly.Xml.textToDom(
+									"<xml>" + blockText + "</xml>"
+								).firstChild;
+								block.setAttribute("gap", 5);
+								xmlList.push(block);
+							}
+						}
+						return xmlList;
+					};
 
-          workspace.registerToolboxCategoryCallback(
-            "searchBlockCategory",
-            editor_blockly.searchBlockCategoryCallback
-          );
+					workspace.registerToolboxCategoryCallback(
+						"entranceCategory",
+						editor_blockly.entranceCategoryCallback
+					);
 
-          var onresize = function (e) {
-            blocklyDiv.style.width = blocklyArea.offsetWidth + "px";
-            blocklyDiv.style.height = blocklyArea.offsetHeight + "px";
-            Blockly.svgResize(workspace);
-          };
-          if (typeof editor !== "undefined" && !editor.isMobile)
-            window.addEventListener("resize", onresize, false);
-          onresize();
-          //Blockly.svgResize(workspace);
+					editor_blockly.searchBlockCategoryCallback = function (workspace) {
+						var xmlList = [];
+						var labels = editor_blockly.searchBlock();
+						for (var i = 0; i < labels.length; i++) {
+							var blockText =
+								"<xml>" + MotaActionBlocks[labels[i]].xmlText() + "</xml>";
+							var block = Blockly.Xml.textToDom(blockText).firstChild;
+							block.setAttribute("gap", 5);
+							xmlList.push(block);
+						}
+						return xmlList;
+					};
 
-          //Blockly.bindEventWithChecks_(workspace.svgGroup_,"wheel",workspace,function(e){});
-          document.getElementById("blocklyDiv").onmousewheel = function (e) {
-            //console.log(e);
-            e.preventDefault();
-            var hvScroll = e.shiftKey ? "hScroll" : "vScroll";
-            var mousewheelOffsetValue =
-              (20 / 380) * workspace.scrollbar[hvScroll].handleLength_ * 3;
-            workspace.scrollbar[hvScroll].handlePosition_ +=
-              (e.deltaY || 0) + (e.detail || 0) > 0
-                ? mousewheelOffsetValue
-                : -mousewheelOffsetValue;
-            workspace.scrollbar[hvScroll].onScroll_();
-            // workspace.setScale(workspace.scale);
-          };
+					workspace.registerToolboxCategoryCallback(
+						"searchBlockCategory",
+						editor_blockly.searchBlockCategoryCallback
+					);
 
-          var doubleClickCheck = [[0, "abc"]];
+					var onresize = function (e) {
+						blocklyDiv.style.width = blocklyArea.offsetWidth + "px";
+						blocklyDiv.style.height = blocklyArea.offsetHeight + "px";
+						Blockly.svgResize(workspace);
+					};
+					if (typeof editor !== "undefined" && !editor.isMobile)
+						window.addEventListener("resize", onresize, false);
+					onresize();
+					//Blockly.svgResize(workspace);
 
-          function omitedcheckUpdateFunction(event) {
-            if (event.type === "create") {
-              editor_blockly.addIntoLastUsedType(event.blockId);
-            }
-            if (event.type === "ui" && event.element == "click") {
-              var newClick = [new Date().getTime(), event.blockId];
-              var lastClick = doubleClickCheck.shift();
-              doubleClickCheck.push(newClick);
-              if (newClick[0] - lastClick[0] < 500) {
-                if (newClick[1] === lastClick[1]) {
-                  editor_blockly.doubleClickBlock(newClick[1]);
-                }
-              }
-            }
-            // Only handle these events
-            if (["create", "move", "change", "delete"].indexOf(event.type) < 0)
-              return;
-            if (editor_blockly.workspace.topBlocks_.length >= 2) {
-              editor_blockly.setValue("入口方块只能有一个");
-              return;
-            }
-            var eventType = editor_blockly.entryType;
-            if (editor_blockly.workspace.topBlocks_.length == 1) {
-              var blockType = editor_blockly.workspace.topBlocks_[0].type;
-              if (
-                blockType !== eventType + "_m" &&
-                !(editor_blockly.isCommonEntry() && blockType == "common_m")
-              ) {
-                editor_blockly.setValue("入口方块类型错误");
-                return;
-              }
-            }
-            try {
-              var code = Blockly.JavaScript.workspaceToCode(workspace).replace(
-                /\\(i|c|d|e|g|z)/g,
-                "\\\\$1"
-              );
-              editor_blockly.setValue(code);
-            } catch (error) {
-              editor_blockly.setValue(String(error));
-              if (error instanceof OmitedError) {
-                var blockName = error.blockName;
-                var varName = error.varName;
-                var block = error.block;
-              }
-              // console.log(error);
-            }
-          }
+					//Blockly.bindEventWithChecks_(workspace.svgGroup_,"wheel",workspace,function(e){});
+					document.getElementById("blocklyDiv").onmousewheel = function (e) {
+						//console.log(e);
+						e.preventDefault();
+						var hvScroll = e.shiftKey ? "hScroll" : "vScroll";
+						var mousewheelOffsetValue =
+							(20 / 380) * workspace.scrollbar[hvScroll].handleLength_ * 3;
+						workspace.scrollbar[hvScroll].handlePosition_ +=
+							(e.deltaY || 0) + (e.detail || 0) > 0 ?
+							mousewheelOffsetValue :
+							-mousewheelOffsetValue;
+						workspace.scrollbar[hvScroll].onScroll_();
+						// workspace.setScale(workspace.scale);
+					};
 
-          workspace.addChangeListener(omitedcheckUpdateFunction);
+					var doubleClickCheck = [
+						[0, "abc"]
+					];
 
-          workspace.addChangeListener(Blockly.Events.disableOrphans);
+					function omitedcheckUpdateFunction(event) {
+						if (event.type === "create") {
+							editor_blockly.addIntoLastUsedType(event.blockId);
+						}
+						if (event.type === "ui" && event.element == "click") {
+							var newClick = [new Date().getTime(), event.blockId];
+							var lastClick = doubleClickCheck.shift();
+							doubleClickCheck.push(newClick);
+							if (newClick[0] - lastClick[0] < 500) {
+								if (newClick[1] === lastClick[1]) {
+									editor_blockly.doubleClickBlock(newClick[1]);
+								}
+							}
+						}
+						// Only handle these events
+						if (["create", "move", "change", "delete"].indexOf(event.type) < 0)
+							return;
+						if (editor_blockly.workspace.topBlocks_.length >= 2) {
+							editor_blockly.setValue("入口方块只能有一个");
+							return;
+						}
+						var eventType = editor_blockly.entryType;
+						if (editor_blockly.workspace.topBlocks_.length == 1) {
+							var blockType = editor_blockly.workspace.topBlocks_[0].type;
+							if (
+								blockType !== eventType + "_m" &&
+								!(editor_blockly.isCommonEntry() && blockType == "common_m")
+							) {
+								editor_blockly.setValue("入口方块类型错误");
+								return;
+							}
+						}
+						try {
+							var code = Blockly.JavaScript.workspaceToCode(workspace).replace(
+								/\\(i|c|d|e|g|z)/g,
+								"\\\\$1"
+							);
+							editor_blockly.setValue(code);
+						} catch (error) {
+							editor_blockly.setValue(String(error));
+							if (error instanceof OmitedError) {
+								var blockName = error.blockName;
+								var varName = error.varName;
+								var block = error.block;
+							}
+							// console.log(error);
+						}
+					}
 
-          editor_blockly.workspace = workspace;
+					workspace.addChangeListener(omitedcheckUpdateFunction);
 
-          MotaActionFunctions.workspace = function () {
-            return editor_blockly.workspace;
-          };
+					workspace.addChangeListener(Blockly.Events.disableOrphans);
 
-          // 因为在editor_blockly.parse里已经HTML转义过一次了,所以这里要覆盖掉以避免在注释中出现<等
-          MotaActionFunctions.xmlText = function (
-            ruleName,
-            inputs,
-            isShadow,
-            comment,
-            collapsed,
-            disabled
-          ) {
-            var rule = MotaActionBlocks[ruleName];
-            var blocktext = isShadow ? "shadow" : "block";
-            var xmlText = [];
-            xmlText.push(
-              "<" +
-                blocktext +
-                ' type="' +
-                ruleName +
-                '"' +
-                (collapsed ? ' collapsed="true"' : "") +
-                (disabled ? ' disabled="true"' : "") +
-                ">"
-            );
-            if (!inputs) inputs = [];
-            for (var ii = 0, inputType; (inputType = rule.argsType[ii]); ii++) {
-              var input = inputs[ii];
-              var _input = "";
-              var noinput = input === null || input === undefined;
-              if (
-                noinput &&
-                inputType === "field" &&
-                MotaActionBlocks[rule.argsGrammarName[ii]].type !==
-                  "field_dropdown"
-              )
-                continue;
-              if (noinput && inputType === "field") {
-                noinput = false;
-                input = rule.fieldDefault(rule.args[ii]);
-              }
-              if (noinput) input = "";
-              if (
-                inputType === "field" &&
-                MotaActionBlocks[rule.argsGrammarName[ii]].type ===
-                  "field_checkbox"
-              )
-                input = input ? "TRUE" : "FALSE";
-              if (inputType !== "field") {
-                var subList = false;
-                var subrulename = rule.argsGrammarName[ii];
-                var subrule = MotaActionBlocks[subrulename];
-                if (subrule instanceof Array) {
-                  subrulename = subrule[subrule.length - 1];
-                  subrule = MotaActionBlocks[subrulename];
-                  subList = true;
-                }
-                _input = subrule.xmlText([], true);
-                if (noinput && !subList && !isShadow) {
-                  //无输入的默认行为是: 如果语句块的备选方块只有一个,直接代入方块
-                  input = subrule.xmlText();
-                }
-              }
-              xmlText.push("<" + inputType + ' name="' + rule.args[ii] + '">');
-              xmlText.push(_input + input);
-              xmlText.push("</" + inputType + ">");
-            }
-            if (comment) {
-              xmlText.push("<comment>");
-              xmlText.push(comment);
-              xmlText.push("</comment>");
-            }
-            var next = inputs[rule.args.length];
-            if (next) {
-              //next
-              xmlText.push("<next>");
-              xmlText.push(next);
-              xmlText.push("</next>");
-            }
-            xmlText.push("</" + blocktext + ">");
-            return xmlText.join("");
-          };
-        })();
+					editor_blockly.workspace = workspace;
 
-        // end mark sfergsvae
-      }
-        .toString()
-        .split("// start mark sfergsvae")[1]
-        .split("// end mark sfergsvae")[0];
-    }
-  },
+					MotaActionFunctions.workspace = function () {
+						return editor_blockly.workspace;
+					};
+
+					// 因为在editor_blockly.parse里已经HTML转义过一次了,所以这里要覆盖掉以避免在注释中出现<等
+					MotaActionFunctions.xmlText = function (
+						ruleName,
+						inputs,
+						isShadow,
+						comment,
+						collapsed,
+						disabled
+					) {
+						var rule = MotaActionBlocks[ruleName];
+						var blocktext = isShadow ? "shadow" : "block";
+						var xmlText = [];
+						xmlText.push(
+							"<" +
+							blocktext +
+							' type="' +
+							ruleName +
+							'"' +
+							(collapsed ? ' collapsed="true"' : "") +
+							(disabled ? ' disabled="true"' : "") +
+							">"
+						);
+						if (!inputs) inputs = [];
+						for (var ii = 0, inputType;
+							(inputType = rule.argsType[ii]); ii++) {
+							var input = inputs[ii];
+							var _input = "";
+							var noinput = input === null || input === undefined;
+							if (
+								noinput &&
+								inputType === "field" &&
+								MotaActionBlocks[rule.argsGrammarName[ii]].type !==
+								"field_dropdown"
+							)
+								continue;
+							if (noinput && inputType === "field") {
+								noinput = false;
+								input = rule.fieldDefault(rule.args[ii]);
+							}
+							if (noinput) input = "";
+							if (
+								inputType === "field" &&
+								MotaActionBlocks[rule.argsGrammarName[ii]].type ===
+								"field_checkbox"
+							)
+								input = input ? "TRUE" : "FALSE";
+							if (inputType !== "field") {
+								var subList = false;
+								var subrulename = rule.argsGrammarName[ii];
+								var subrule = MotaActionBlocks[subrulename];
+								if (subrule instanceof Array) {
+									subrulename = subrule[subrule.length - 1];
+									subrule = MotaActionBlocks[subrulename];
+									subList = true;
+								}
+								_input = subrule.xmlText([], true);
+								if (noinput && !subList && !isShadow) {
+									//无输入的默认行为是: 如果语句块的备选方块只有一个,直接代入方块
+									input = subrule.xmlText();
+								}
+							}
+							xmlText.push("<" + inputType + ' name="' + rule.args[ii] + '">');
+							xmlText.push(_input + input);
+							xmlText.push("</" + inputType + ">");
+						}
+						if (comment) {
+							xmlText.push("<comment>");
+							xmlText.push(comment);
+							xmlText.push("</comment>");
+						}
+						var next = inputs[rule.args.length];
+						if (next) {
+							//next
+							xmlText.push("<next>");
+							xmlText.push(next);
+							xmlText.push("</next>");
+						}
+						xmlText.push("</" + blocktext + ">");
+						return xmlText.join("");
+					};
+				})();
+
+				// end mark sfergsvae
+			}
+			.toString()
+			.split("// start mark sfergsvae")[1]
+			.split("// end mark sfergsvae")[0];
+	}
+},
     "夹击激光动画": function () {
     function createCanvas(name, zIndex) {
       if (!name) return;
@@ -12461,2035 +13274,2079 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
     };
   },
     "小地图": function () {
-    // 在此增加新插件
-    // ----- 不可自定义 杂七杂八的变量
-    /** @type {{[x: string]: BFSResult}} */
-    let mapCache = {}; // 地图缓存
-    let drawCache = {}; // 绘制信息缓存
-    let status = "none"; // 当前的绘制状态
-    /** @type {{[x: string]: Sprite}} */
-    let sprites = {}; // 当前所有的sprite
-    /** @type {{[x: string]: Sprite}} */
-    let canDrag = {}; // 可以拖拽的sprite
-    /** @type {{[x: string]: Button}} */
-    let areaSprite = {}; // 区域列表对应的sprite
-    let clicking = false; // 是否正在点击,用于拖拽判定
-    let drawingMap = ""; // 正在绘制的中心楼层
-    let nowScale = 0; // 当前绘制的放缩比例
-    let lastTouch = {}; // 上一次的单点点击信息
-    let lastLength = 0; // 手机端缩放时上一次的两指间距离
-    let nowDepth = 0; // 当前的遍历深度
-    let drawedThumbnail = {}; // 已经绘制过的缩略图
-    let moved = false; // 鼠标按下后是否移动了
-    let noBorder = false; // 是否是无边框拼接模式
-    let lastScale = 0; // 上一次缩放,用于优化缩略图绘制
-    let areaPage = 0; // 区域显示的当前页数
-    let nowArea = 0; // 当前区域index
-    let selecting = ""; // 选择时当前正在选择的地图
+	// 在此增加新插件
+	// ----- 不可自定义 杂七杂八的变量
+	/** @type {{[x: string]: BFSResult}} */
+	let mapCache = {}; // 地图缓存
+	let drawCache = {}; // 绘制信息缓存
+	let status = "none"; // 当前的绘制状态
+	/** @type {{[x: string]: Sprite}} */
+	let sprites = {}; // 当前所有的sprite
+	/** @type {{[x: string]: Sprite}} */
+	let canDrag = {}; // 可以拖拽的sprite
+	/** @type {{[x: string]: Button}} */
+	let areaSprite = {}; // 区域列表对应的sprite
+	let clicking = false; // 是否正在点击,用于拖拽判定
+	let drawingMap = ""; // 正在绘制的中心楼层
+	let nowScale = 0; // 当前绘制的放缩比例
+	let lastTouch = {}; // 上一次的单点点击信息
+	let lastLength = 0; // 手机端缩放时上一次的两指间距离
+	let nowDepth = 0; // 当前的遍历深度
+	let drawedThumbnail = {}; // 已经绘制过的缩略图
+	let moved = false; // 鼠标按下后是否移动了
+	let noBorder = false; // 是否是无边框拼接模式
+	let lastScale = 0; // 上一次缩放,用于优化缩略图绘制
+	let areaPage = 0; // 区域显示的当前页数
+	let nowArea = 0; // 当前区域index
+	let selecting = ""; // 选择时当前正在选择的地图
 
-    // ---- 不可自定义,常量
-    /** @type {Area} */
-    let areas = []; // 区域信息
-    const perPage = Math.floor((core._PY_ - 60) / 30); // 区域的每页显示数量
+	// ---- 不可自定义,常量
+	/** @type {Area} */
+	let areas = []; // 区域信息
+	const perPage = Math.floor((core._PY_ - 60) / 30); // 区域的每页显示数量
 
-    // ---- 可自定义,默认的切换地图的图块id
-    const defaultChange = {
-      left: "leftPortal", // 左箭头
-      up: "upPortal", // 上箭头
-      right: "rightPortal", // 右箭头
-      down: "downPortal", // 下箭头
-      upFloor: "upFloor", // 上楼
-      downFloor: "downFloor", // 下楼
-    };
-    // ---- 可自定义,默认数值
-    const defaultValue = {
-      font: "Verdana", // 默认字体
-      scale: 60, // 默认地图缩放比例
-      depth: Infinity, // 默认的遍历深度
-    };
+	// ---- 可自定义,默认的切换地图的图块id
+	const defaultChange = {
+		left: "leftPortal", // 左箭头
+		up: "upPortal", // 上箭头
+		right: "rightPortal", // 右箭头
+		down: "downPortal", // 下箭头
+		upFloor: "upFloor", // 上楼
+		downFloor: "downFloor", // 下楼
+	};
+	// ---- 可自定义,默认数值
+	const defaultValue = {
+		font: "Verdana", // 默认字体
+		scale: 60, // 默认地图缩放比例
+		depth: Infinity, // 默认的遍历深度
+	};
 
-    // ---- 不可自定义,计算数据
-    const dirData = {
-      up: [1, 0],
-      down: [-1, 0],
-      left: [0, 1],
-      right: [0, -1],
-      upFloor: [0, 0],
-      downFloor: [0, 0],
-    };
-    let ignoreEnemies = (this.ignoreEnemies = []);
+	// ---- 不可自定义,计算数据
+	const dirData = {
+		up: [1, 0],
+		down: [-1, 0],
+		left: [0, 1],
+		right: [0, -1],
+		upFloor: [0, 0],
+		downFloor: [0, 0],
+	};
+	let ignoreEnemies = (this.ignoreEnemies = []);
 
-    let allChangeEntries = Object.entries(defaultChange);
+	let allChangeEntries = Object.entries(defaultChange);
 
-    this.setq = function (floorId) {
-      core.setFlag("任务地点", floorId);
-    };
+	this.setq = function (floorId) {
+		core.setFlag("任务地点", floorId);
+	};
 
-    const reset = core.events.resetGame;
-    this.bfs = function () {
-      areas = [];
-      // 获取所有分区,使用异步函数,保证不会卡顿
-      // 原理是用bfs扫,将所有连在一起的地图合并成一个区域
-      (async function () {
-        let all = core.floorIds.slice();
-        const scanned = {
-          [all[0]]: true,
-        };
-        while (all.length > 0) {
-          let now = all.shift();
-          if (core.status.maps[now].deleted) continue;
-          if (!now) return;
-          await new Promise((res) => {
-            const result = core.plugin.bfsSearch(now, Infinity, true);
-            mapCache[`${now}_Infinity_false`] = result;
-            areas.push({ name: core.floors[now].areas, maps: result.order });
-            for (const map of result.order) {
-              scanned[map] = true;
-              all = all.filter((v) => !result.order.includes(v));
-            }
-            res("success");
-          }).then(() => {
-            core.setFlag("areas", areas);
-          });
-        }
-      })();
-    };
-    core.events.resetGame = function () {
-      reset.apply(core.events, arguments);
-      core.plugin.bfs();
-    };
-    /**
-     * 广度优先搜索搜索地图路径
-     * @param {string} center 中心地图的id
-     * @param {number} depth 搜索深度
-     * @param {boolean} noCache 是否不使用缓存
-     * @returns {BFSResult} 格式:floorId_x_y_dir: floorId_x_y
-     */
-    this.bfsSearch = function bfsSearch(center, depth, noCache) {
-      // 检查缓存
-      const id = `${center}_${depth}_${noBorder}`;
-      if (mapCache[id] && !noCache) return mapCache[id];
-      const used = {
-        [center]: true,
-      }; // 搜索过的楼层
-      let queue = [];
-      let stack = [center]; // 当前栈
-      let nowDepth = -1;
-      const mapOrder = [center]; // 遍历顺序,顺便还能记录遍历了哪些楼层
+	const reset = core.events.resetGame;
+	this.bfs = function () {
+		areas = [];
+		// 获取所有分区,使用异步函数,保证不会卡顿
+		// 原理是用bfs扫,将所有连在一起的地图合并成一个区域
+		(async function () {
+			let all = core.floorIds.slice();
+			const scanned = {
+				[all[0]]: true,
+			};
+			while (all.length > 0) {
+				let now = all.shift();
+				if (core.status.maps[now].deleted) continue;
+				if (!now) return;
+				await new Promise((res) => {
+					const result = core.plugin.bfsSearch(now, Infinity, true);
+					mapCache[`${now}_Infinity_false`] = result;
+					areas.push({ name: core.floors[now].areas, maps: result.order });
+					for (const map of result.order) {
+						scanned[map] = true;
+						all = all.filter((v) => !result.order.includes(v));
+					}
+					res("success");
+				}).then(() => {
+					core.setFlag("areas", areas);
+				});
+			}
+		})();
+	};
+	core.events.resetGame = function () {
+		reset.apply(core.events, arguments);
+		core.plugin.bfs();
+	};
+	/**
+	 * 广度优先搜索搜索地图路径
+	 * @param {string} center 中心地图的id
+	 * @param {number} depth 搜索深度
+	 * @param {boolean} noCache 是否不使用缓存
+	 * @returns {BFSResult} 格式:floorId_x_y_dir: floorId_x_y
+	 */
+	this.bfsSearch = function bfsSearch(center, depth, noCache) {
+		// 检查缓存
+		const id = `${center}_${depth}_${noBorder}`;
+		if (mapCache[id] && !noCache) return mapCache[id];
+		const used = {
+			[center]: true,
+		}; // 搜索过的楼层
+		let queue = [];
+		let stack = [center]; // 当前栈
+		let nowDepth = -1;
+		const mapOrder = [center]; // 遍历顺序,顺便还能记录遍历了哪些楼层
 
-      const res = {}; // 输出结果,格式:floorId_x_y_dir: floorId_x_y
-      const enemies = {};
-      const upOrDown = {};
-      const mapdir = {};
-      // 开始循环搜索
-      while (nowDepth < depth && stack.length > 0) {
-        const now = stack.shift(); // 当前id
-        if (core.status.maps[now].deleted) continue;
-        mapdir[now] = mapdir[now] ?? [];
-        const blocks = core.getMapBlocksObj(now); // 获取当前地图的每点的事件
-        enemies[now] = {};
-        // 遍历,获取可以传送的点,只检测绿点事件,因此可用红点事件进行传送来实现分区功能
-        for (const i in blocks) {
-          const block = blocks[i];
-          // 整合漏怪检测,所以要检测怪物
-          if (block.event.trigger === "battle") {
-            const id = block.event.id;
-            if (ignoreEnemies.includes(id)) continue;
-            else enemies[now][i] = block.event.id;
-            continue;
-          }
-          // 检测触发器是否为切换楼层,不是则直接跳过
-          if (block.event.trigger !== "changeFloor") continue;
-          const dirEntries = allChangeEntries.find(
-            (v) => v[1] === block.event.id
-          );
-          // 如果不是那六种传送门,直接忽略
-          if (!dirEntries) continue;
-          const data = block.event.data;
-          const dir = dirEntries[0];
-          const route = now + "_" + i.replace(",", "_") + "_" + dir;
-          const target = data.floorId + "_" + data.loc.join("_");
+		const res = {}; // 输出结果,格式:floorId_x_y_dir: floorId_x_y
+		const enemies = {};
+		const upOrDown = {};
+		const mapdir = {};
+		// 开始循环搜索
+		while (nowDepth < depth && stack.length > 0) {
+			const now = stack.shift(); // 当前id
+			if (core.status.maps[now].deleted) continue;
+			mapdir[now] = mapdir[now] ?? [];
+			const blocks = core.getMapBlocksObj(now); // 获取当前地图的每点的事件
+			enemies[now] = {};
+			// 遍历,获取可以传送的点,只检测绿点事件,因此可用红点事件进行传送来实现分区功能
+			for (const i in blocks) {
+				const block = blocks[i];
+				// 整合漏怪检测,所以要检测怪物
+				if (block.event.trigger === "battle") {
+					const id = block.event.id;
+					if (ignoreEnemies.includes(id)) continue;
+					else enemies[now][i] = block.event.id;
+					continue;
+				}
+				// 检测触发器是否为切换楼层,不是则直接跳过
+				if (block.event.trigger !== "changeFloor") continue;
+				const dirEntries = allChangeEntries.find(
+					(v) => v[1] === block.event.id
+				);
+				// 如果不是那六种传送门,直接忽略
+				if (!dirEntries) continue;
+				const data = block.event.data;
+				const dir = dirEntries[0];
+				const route = now + "_" + i.replace(",", "_") + "_" + dir;
+				const target = data.floorId + "_" + data.loc.join("_");
 
-          mapdir[now].push(dir);
-          if (!used[data.floorId]) {
-            if (dir === "upFloor" || dir === "downFloor") {
-              upOrDown[now] = upOrDown[id] ?? [];
-              upOrDown[now].push(dir);
-            }
+				mapdir[now].push(dir);
+				if (!used[data.floorId]) {
+					if (dir === "upFloor" || dir === "downFloor") {
+						upOrDown[now] = upOrDown[id] ?? [];
+						upOrDown[now].push(dir);
+					}
 
-            queue.push(data.floorId); // 没有搜索过,则加入栈中
-            mapOrder.push(data.floorId);
-            used[data.floorId] = true;
-          }
-          res[route] = target;
-        }
-        if (stack.length === 0) {
-          stack = queue;
-          queue = [];
-          nowDepth++;
-        }
-        if (stack.length === 0 && queue.length === 0) break;
-      }
-      return { res, order: mapOrder, enemies, upOrDown, mapdir };
-    };
-    /**
-     * 获取绘制信息
-     * @param {string?} center 中心地图id
-     * @param {number?} depth 搜索深度
-     * @param {boolean?} noCache 是否不使用缓存
-     * @returns {MapDrawInfo}
-     */
-    this.getMapDrawInfo = function (
-      center = core.status.floorId,
-      depth = defaultValue.depth,
-      noCache = false
-    ) {
-      nowDepth = depth;
-      drawingMap = center;
-      const id = `${center}_${depth}_${noBorder}`;
-      // 检查缓存
-      if (drawCache[id] && !noCache) return drawCache[id];
-      const map = core.plugin.bfsSearch(center, depth, noCache);
-      mapCache[id] = map;
-      const res = getDrawInfo(map.res, center, map.order);
-      res.upOrDown = map.upOrDown;
-      res.mapdir = map.mapdir;
-      drawCache[id] = res;
-      return res;
-    };
-    /**
-     * 提供地图的绘制信息
-     * @param {{[x: string]: string}} map 要绘制的地图,格式:floorId_x_y_dir: floorId_x_y
-     * @param {string} center 中心地图的id
-     * @param {string[]} order 遍历顺序
-     * @returns {MapDrawInfo} 地图的绘制信息
-     */
+					queue.push(data.floorId); // 没有搜索过,则加入栈中
+					mapOrder.push(data.floorId);
+					used[data.floorId] = true;
+				}
+				res[route] = target;
+			}
+			if (stack.length === 0) {
+				stack = queue;
+				queue = [];
+				nowDepth++;
+			}
+			if (stack.length === 0 && queue.length === 0) break;
+		}
+		return { res, order: mapOrder, enemies, upOrDown, mapdir };
+	};
+	/**
+	 * 获取绘制信息
+	 * @param {string?} center 中心地图id
+	 * @param {number?} depth 搜索深度
+	 * @param {boolean?} noCache 是否不使用缓存
+	 * @returns {MapDrawInfo}
+	 */
+	this.getMapDrawInfo = function (
+		center = core.status.floorId,
+		depth = defaultValue.depth,
+		noCache = false
+	) {
+		nowDepth = depth;
+		drawingMap = center;
+		const id = `${center}_${depth}_${noBorder}`;
+		// 检查缓存
+		if (drawCache[id] && !noCache) return drawCache[id];
+		const map = core.plugin.bfsSearch(center, depth, noCache);
+		mapCache[id] = map;
+		const res = getDrawInfo(map.res, center, map.order);
+		res.upOrDown = map.upOrDown;
+		res.mapdir = map.mapdir;
+		drawCache[id] = res;
+		return res;
+	};
+	/**
+	 * 提供地图的绘制信息
+	 * @param {{[x: string]: string}} map 要绘制的地图,格式:floorId_x_y_dir: floorId_x_y
+	 * @param {string} center 中心地图的id
+	 * @param {string[]} order 遍历顺序
+	 * @returns {MapDrawInfo} 地图的绘制信息
+	 */
 
-    function getDrawInfo(map, center, order) {
-      // 先根据地图id分类,从而确定每个地图连接哪些地图,同时方便处理
-      const links = {};
-      for (const i in map) {
-        const splitted = i.split("_");
-        const id = splitted[0];
-        if (!links[id]) links[id] = {};
-        links[id][i] = map[i];
-      }
+	function getDrawInfo(map, center, order) {
+		// 先根据地图id分类,从而确定每个地图连接哪些地图,同时方便处理
+		const links = {};
+		for (const i in map) {
+			const splitted = i.split("_");
+			const id = splitted[0];
+			if (!links[id]) links[id] = {};
+			links[id][i] = map[i];
+		}
 
-      // 分类完毕,然后根据连接点先计算出各个地图的坐标,然后再进行判断
-      const centerFloor = core.status.maps[center];
-      const visitedCenter = core.hasVisitedFloor(center);
-      const locs = {
-        // 格式:[中心x, 中心y, 宽, 高, 是否到达过]
-        [center]: [2, 2, 1, 1, visitedCenter],
-      };
-      // 可以上楼下楼的地图
-      const upOrDown = {};
-      for (const id of order) {
-        const now = links[id];
-        // 遍历每一个地图的连接情况
-        for (const from in now) {
-          const to = now[from];
-          // 先根据from to计算物理位置
-          const fromData = from.split("_"),
-            toData = to.split("_");
-          const dir = fromData[3];
-          if (dir === "upFloor" || dir === "downFloor") continue;
-          if (!defaultChange[dir]) continue;
-          const v = dirData[dir][1], // 竖直数值
-            h = dirData[dir][0], // 水平数值
-            ha = Math.abs(h),
-            va = Math.abs(v);
-          const ff = id, // fromFloorId
-            tf = toData[0]; // toFloorId
-          const fromFloor = core.status.maps[ff],
-            toFloor = core.status.maps[tf];
-          const fhw = Math.floor(fromFloor.width / 2), // fromFloorHalfWidth
-            fhh = Math.floor(fromFloor.height / 2),
-            thw = Math.floor(toFloor.width / 2),
-            thh = Math.floor(toFloor.height / 2);
-          const fLoc = locs[id] ?? [0, 0];
-          if (!locs[ff]) continue;
-          let x, y;
-          if (locs && locs[tf]) {
-            x = locs[tf][0];
-            y = locs[tf][1];
-          } else {
-            // 计算坐标,公式可以通过画图推断出
-            x = fLoc[0] - v;
-            y = fLoc[1] - h;
-          }
-          locs[tf] = locs[tf] ?? [x, y, 1, 1, core.hasVisitedFloor(tf)];
-        }
-      }
-      // 获取地图绘制需要的长宽
-      let width = 0,
-        height = 0;
-      let left, right, up, down;
-      for (const id in locs) {
-        const [x, y, w, h] = locs[id];
-        if (left === void 0) {
-          left = right = x;
-          up = down = y;
-        }
-        left = Math.min(x - 1, left);
-        right = Math.max(x + 1, right);
-        up = Math.min(y - 1, up);
-        down = Math.max(y + 1, down);
-      }
-      width = right - left;
-      height = down - up;
+		// 分类完毕,然后根据连接点先计算出各个地图的坐标,然后再进行判断
+		const centerFloor = core.status.maps[center];
+		const visitedCenter = core.hasVisitedFloor(center);
+		const locs = {
+			// 格式:[中心x, 中心y, 宽, 高, 是否到达过]
+			[center]: [2, 2, 1, 1, visitedCenter],
+		};
+		// 可以上楼下楼的地图
+		const upOrDown = {};
+		for (const id of order) {
+			const now = links[id];
+			// 遍历每一个地图的连接情况
+			for (const from in now) {
+				const to = now[from];
+				// 先根据from to计算物理位置
+				const fromData = from.split("_"),
+					toData = to.split("_");
+				const dir = fromData[3];
+				if (dir === "upFloor" || dir === "downFloor") continue;
+				if (!defaultChange[dir]) continue;
+				const v = dirData[dir][1], // 竖直数值
+					h = dirData[dir][0], // 水平数值
+					ha = Math.abs(h),
+					va = Math.abs(v);
+				const ff = id, // fromFloorId
+					tf = toData[0]; // toFloorId
+				const fromFloor = core.status.maps[ff],
+					toFloor = core.status.maps[tf];
+				const fhw = Math.floor(fromFloor.width / 2), // fromFloorHalfWidth
+					fhh = Math.floor(fromFloor.height / 2),
+					thw = Math.floor(toFloor.width / 2),
+					thh = Math.floor(toFloor.height / 2);
+				const fLoc = locs[id] ?? [0, 0];
+				if (!locs[ff]) continue;
+				let x, y;
+				if (locs && locs[tf]) {
+					x = locs[tf][0];
+					y = locs[tf][1];
+				} else {
+					// 计算坐标,公式可以通过画图推断出
+					x = fLoc[0] - v;
+					y = fLoc[1] - h;
+				}
+				locs[tf] = locs[tf] ?? [x, y, 1, 1, core.hasVisitedFloor(tf)];
+			}
+		}
+		// 获取地图绘制需要的长宽
+		let width = 0,
+			height = 0;
+		let left, right, up, down;
+		for (const id in locs) {
+			const [x, y, w, h] = locs[id];
+			if (left === void 0) {
+				left = right = x;
+				up = down = y;
+			}
+			left = Math.min(x - 1, left);
+			right = Math.max(x + 1, right);
+			up = Math.min(y - 1, up);
+			down = Math.max(y + 1, down);
+		}
+		width = right - left;
+		height = down - up;
 
-      return { locs, width, height, layer: upOrDown };
-    }
+		return { locs, width, height, layer: upOrDown };
+	}
 
-    function mapblock(mapdir) {
-      let mb = "";
-      if (mapdir.includes("up")) mb += "u";
-      if (mapdir.includes("down")) mb += "d";
-      if (mapdir.includes("left")) mb += "l";
-      if (mapdir.includes("right")) mb += "r";
-      return mb ? mb + ".webp" : "null.webp";
-    }
-    core.animateFrame.globalAlphaFloor = 0;
-    core.animateFrame.globalAlphaFloorStatus = 1;
+	function mapblock(mapdir) {
+		let mb = "";
+		if (mapdir.includes("up")) mb += "u";
+		if (mapdir.includes("down")) mb += "d";
+		if (mapdir.includes("left")) mb += "l";
+		if (mapdir.includes("right")) mb += "r";
+		return mb ? mb + ".webp" : "null.webp";
+	}
+	core.animateFrame.globalAlphaFloor = 0;
+	core.animateFrame.globalAlphaFloorStatus = 1;
 
-    const tesk = document.createElement("canvas");
-    tesk.width = 300;
-    tesk.height = 300;
-    const teskctx = tesk.getContext("2d");
+	const tesk = document.createElement("canvas");
+	tesk.width = 300;
+	tesk.height = 300;
+	const teskctx = tesk.getContext("2d");
 
-    let line = 50;
-    teskctx.strokeStyle = "green";
-    teskctx.fillStyle = "green";
-    let now = 0;
-    core.registerAnimationFrame("tesk", true, function (timestamp) {
-      if (timestamp - now > 1000 / 60) {
-        now = timestamp;
-        core.clearMap(teskctx);
-        teskctx.lineWidth = 150 - line;
+	let line = 50;
+	teskctx.strokeStyle = "green";
+	teskctx.fillStyle = "green";
+	let now = 0;
+	core.registerAnimationFrame("tesk", true, function (timestamp) {
+		if (timestamp - now > 1000 / 60) {
+			now = timestamp;
+			core.clearMap(teskctx);
+			teskctx.lineWidth = 150 - line;
 
-        if (line <= 150) {
-          teskctx.beginPath();
-          teskctx.arc(150, 150, line, 0, Math.PI * 2);
-          line += 2;
-          teskctx.stroke();
-        } else {
-          teskctx.beginPath();
-          teskctx.arc(150, 150, line - 150, 0, Math.PI * 2);
-          line += 2;
-          teskctx.fill();
-          if (line >= 250) line = 50;
-        }
-      }
-    });
+			if (line <= 150) {
+				teskctx.beginPath();
+				teskctx.arc(150, 150, line, 0, Math.PI * 2);
+				line += 2;
+				teskctx.stroke();
+			} else {
+				teskctx.beginPath();
+				teskctx.arc(150, 150, line - 150, 0, Math.PI * 2);
+				line += 2;
+				teskctx.fill();
+				if (line >= 250) line = 50;
+			}
+		}
+	});
 
-    /**
-     * 绘制小地图
-     * @param {MapDrawInfo} info 地图绘制信息
-     * @param {number} scale 地图的绘制比例
-     */
-    this.drawSmallMap = function (
-      ctx,
-      info,
-      center,
-      sx,
-      sy,
-      sw,
-      sh,
-      scale = defaultValue.scale
-    ) {
-      core.clearMap(ctx, sx, sy, sw + 40, sh + 60);
-      if (core.domStyle.isVertical) {
-        sy += 50;
-        sx += 15;
-      } else {
-        sy += 60;
-        sx += 30;
-      }
-      core.fillRect(ctx, sx, sy, sw, sh, "#000");
-      core.strokeRect(ctx, sx, sy, sw, sh, "#fff", 5);
-      core.setTextAlign("outerUI", "center");
-      core.fillBoldText1(
-        ctx,
-        core.status.maps[center].areas,
-        sx + sw / 2,
-        sy - 10,
-        "#FFFFFF",
-        "#000000",
-        6,
-        "bold 42px Verdana"
-      );
-      const locs = info.locs;
-      for (const id in locs) {
-        const loc = locs[id];
-        let color = "#000";
-        if (!loc[4]) color = "#f0f";
-        const [x, y, w, h] = loc.map((v) => typeof v === "number" && v * scale);
-        const fx = x + sx,
-          fy = y + sy;
-        const mapdir = info.mapdir[id];
-        const img = mapblock(mapdir);
-        if (x < 0 || x > 4 * scale || y < 0 || y > 4 * scale) continue;
-        core.drawImage(ctx, img, 0, 0, 60, 60, fx, fy, w, h);
-        const layer = info.upOrDown[id];
-        const min = Math.min(w, h);
-        if (core.getFlag("任务地点") && core.getFlag("任务地点") === id)
-          ctx.drawImage(tesk, fx + min / 4, fy + min / 4, min / 2, min / 2);
-        if (layer?.includes("upFloor"))
-          core.drawIcon(
-            ctx,
-            defaultChange.upFloor,
-            fx + min / 4,
-            fy + min / 4,
-            min / 2,
-            min / 2
-          );
-        if (layer?.includes("downFloor"))
-          core.drawIcon(
-            ctx,
-            defaultChange.downFloor,
-            fx + min / 4,
-            fy + min / 4,
-            min / 2,
-            min / 2
-          );
+	/**
+	 * 绘制小地图
+	 * @param {MapDrawInfo} info 地图绘制信息
+	 * @param {number} scale 地图的绘制比例
+	 */
+	this.drawSmallMap = function (
+		ctx,
+		info,
+		center,
+		sx,
+		sy,
+		sw,
+		sh,
+		scale = defaultValue.scale
+	) {
+		core.clearMap(ctx, sx, sy, sw + 40, sh + 60);
+		if (core.domStyle.isVertical) {
+			sy += 50;
+			sx += 15;
+		} else {
+			sy += 60;
+			sx += 30;
+		}
+		core.fillRect(ctx, sx - 10, sy - 10, sw + 20, sh + 20, "#000");
+		core.strokeRect(ctx, sx - 10, sy - 10, sw + 20, sh + 20, "#fff", 5);
+		core.setTextAlign("outerUI", "center");
+		core.fillBoldText1(
+			ctx,
+			core.status.maps[center].areas,
+			sx + sw / 2,
+			sy - 20,
+			"#FFFFFF",
+			"#000000",
+			6,
+			"bold 42px Verdana"
+		);
+		const locs = info.locs;
+		for (const id in locs) {
+			const loc = locs[id];
+			let color = "#000";
+			if (!loc[4]) color = "#f0f";
+			const [x, y, w, h] = loc.map((v) => typeof v === "number" && v * scale);
+			const fx = x + sx,
+				fy = y + sy;
+			const mapdir = info.mapdir[id];
+			const img = mapblock(mapdir);
+			if (x < 0 || x > 4 * scale || y < 0 || y > 4 * scale) continue;
+			core.drawImage(ctx, img, 0, 0, 60, 60, fx, fy, w, h);
+			const layer = info.upOrDown[id];
+			const min = Math.min(w, h);
 
-        // 显示漏怪数量
-        if (core.getFlag("showEnemy")) {
-          ctx.textAlign = "center";
-          ctx.textBaseline = "middle";
-          const c = drawingMap + "_" + nowDepth + "_" + noBorder;
-          const n = Object.keys(mapCache[c].enemies[id]).length;
-          color = "#fff";
-          if (n > 10) color = "#fc3";
-          if (n > 20) color = "#f22";
-          ctx.shadowBlur = 0.6 * nowScale;
-          ctx.shadowColor = "#000";
-          if (n > 0)
-            core.fillText(
-              ctx,
-              n,
-              fx + (w * 3) / 10,
-              fy + (h * 7) / 10,
-              color,
-              22 + "px normal"
-            );
-          ctx.shadowBlur = 0;
-        }
-      }
-    };
-  },
+			if (layer?.includes("upFloor"))
+				core.drawIcon(
+					ctx,
+					defaultChange.upFloor,
+					fx + min / 4,
+					fy + min / 4,
+					min / 2,
+					min / 2
+				);
+			if (layer?.includes("downFloor"))
+				core.drawIcon(
+					ctx,
+					defaultChange.downFloor,
+					fx + min / 4,
+					fy + min / 4,
+					min / 2,
+					min / 2
+				);
+			if (core.getFlag("任务地点") && core.getFlag("任务地点") === id)
+				ctx.drawImage(tesk, fx + min / 4, fy + min / 4, min / 2, min / 2);
+			if (id === core.status.floorId)
+				core.drawImage(
+					ctx,
+					"hero.webp",
+					0,
+					0,
+					32,
+					19,
+					fx + min / 4,
+					fy + (min * 5) / 16,
+					32,
+					19
+				);
+			// 显示漏怪数量
+			if (core.getFlag("showEnemy")) {
+				ctx.textAlign = "center";
+				ctx.textBaseline = "middle";
+				const c = drawingMap + "_" + nowDepth + "_" + noBorder;
+				const n = Object.keys(mapCache[c].enemies[id]).length;
+				color = "#fff";
+				if (n > 10) color = "#fc3";
+				if (n > 20) color = "#f22";
+				ctx.shadowBlur = 0.6 * nowScale;
+				ctx.shadowColor = "#000";
+				if (n > 0)
+					core.fillText(
+						ctx,
+						n,
+						fx + (w * 3) / 10,
+						fy + (h * 7) / 10,
+						color,
+						22 + "px normal"
+					);
+				ctx.shadowBlur = 0;
+			}
+			if (!core.hasVisitedFloor(id)) {
+				core.fillRect(ctx, fx, fy, w, h, "rgba(0,0,0,0.7)");
+				core.fillText(
+					ctx,
+					"?",
+					fx + min / 2,
+					fy + (min * 3) / 4,
+					"#FFFFFF",
+					"bold 42px Verdana"
+				);
+			}
+		}
+	};
+},
     "楼传": function () {
-    // 在此增加新插件
+	// 在此增加新插件
 
-    core.canMoveFloor = function () {
-      let canmove = false;
-      core.status.thisMap.blocks.forEach((block) => {
-        if (
-          !block.disable &&
-          (block.event.id == "upFloor" || block.event.id == "downFloor")
-        ) {
-          let automaticRoute = core.automaticRoute(block.x, block.y);
-          if (!core.flags.flyNearStair || automaticRoute.length > 0) {
-            let loc = automaticRoute.pop();
-            loc = automaticRoute.pop();
-            if (core.canMoveDirectly(loc?.x, loc?.y) >= 0 || !loc) {
-              canmove = true;
-            }
-          }
-        }
-      });
-      return canmove;
-    };
-    ui.prototype._drawViewMaps_drawHint = function () {
-      core.playSound("打开界面");
-    };
+	core.canMoveFloor = function () {
+		let canmove = false;
+		core.status.thisMap.blocks.forEach((block) => {
+			if (
+				!block.disable &&
+				(block.event.id == "upFloor" || block.event.id == "downFloor")
+			) {
+				let automaticRoute = core.automaticRoute(block.x, block.y);
+				if (!core.flags.flyNearStair || automaticRoute.length > 0) {
+					let loc = automaticRoute.pop();
+					loc = automaticRoute.pop();
+					if (core.canMoveDirectly(loc?.x, loc?.y) >= 0 || !loc) {
+						canmove = true;
+					}
+				}
+			}
+		});
+		return canmove;
+	};
+	ui.prototype._drawViewMaps_drawHint = function () {
+		core.playSound("打开界面");
+	};
 
-    ////// 绘制浏览地图界面 //////
-    ui.prototype._drawViewMaps = function (index, x, y) {
-      core.lockControl();
-      core.clearMap("data");
-      core.status.event.id = "viewMaps";
-      this.clearUI();
-      //console.log(index)
-      if (index == null) index = core.floorIds.indexOf(core.status.floorId);
-      core.animateFrame.tip = null;
-      core.status.checkBlock.cache = {};
-      let data = this._drawViewMaps_buildData(index, x, y);
-      core.drawWindowSkin("winskin1.webp", "ui", 0, 0, 416, 416);
-      let page = core.status.event.data.index;
-      let floorId = core.status.event.data.floorId;
-      core.ui.statusBar._update_map(floorId);
-      const bfs = core.plugin.bfsSearch(floorId, 1, true);
-      const mapdir = bfs.mapdir[floorId];
-      core.setTextAlign("ui", "center");
-      let size = (core.__PIXELS__ * 3) / 4; //312
-      const areas = core.getFlag("areas");
+	////// 绘制浏览地图界面 //////
+	ui.prototype._drawViewMaps = function (index, x, y) {
+		core.lockControl();
 
-      let i = areas.findIndex((v) => v.maps.includes(floorId));
-      core.fillRoundRect("ui", 15 - 2, 15 - 2, 35 + 4, 35 + 4, 4, "#444444");
-      core.strokeRoundRect(
-        "ui",
-        15 - 4,
-        15 - 4,
-        35 + 8,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      core.fillBoldText1(
-        "ui",
-        "当前",
-        13 + 20,
-        17 + 20,
-        "#FFFFFF",
-        "#000000",
-        2,
-        this._buildFont(18, true)
-      );
+		core.clearMap("data");
+		core.status.event.id = "viewMaps";
+		this.clearUI();
+		//console.log(index)
+		if (index == null) index = core.floorIds.indexOf(core.status.floorId);
+		core.animateFrame.tip = null;
+		core.status.checkBlock.cache = {};
+		let data = this._drawViewMaps_buildData(index, x, y);
+		core.drawWindowSkin("winskin1.webp", "ui", 0, 0, 416, 416);
+		let page = core.status.event.data.index;
+		let floorId = core.status.event.data.floorId;
+		core.ui.statusBar._update_map(floorId);
+		const bfs = core.plugin.bfsSearch(floorId, 1, true);
+		const mapdir = bfs.mapdir[floorId];
+		core.setTextAlign("ui", "center");
+		let size = (core.__PIXELS__ * 3) / 4; //312
+		const areas = core.getFlag("areas");
 
-      core.fillRoundRect(
-        "ui",
-        15 - 2,
-        15 - 2 + 35 + 8 + size + 8 - 54,
-        35 + 4,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4,
-        15 - 4 + 35 + 8 + size + 8 - 54,
-        35 + 8,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (
-        !core.status.maps[core.floorIds[page]].canFlyTo ||
-        !core.hasVisitedFloor(core.floorIds[page])
-      ) {
-        core.fillBoldText1(
-          "ui",
-          "预览",
-          13 + 20,
-          17 + 20 + 35 + 8 + size + 8 - 54,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "传送",
-          13 + 20,
-          17 + 20 + 35 + 8 + size + 8 - 54,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
-      core.fillRoundRect(
-        "ui",
-        15 - 4 + size - 2 + 45,
-        15 - 2 + size - 4 + 45,
-        35 + 4,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4 + size - 4 + 45,
-        15 - 4 + size - 4 + 45,
-        35 + 8,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      core.fillBoldText1(
-        "ui",
-        "离开",
-        15 - 4 + size - 4 + 45 + 22,
-        15 - 4 + size - 4 + 45 + 26,
-        "#FFFFFF",
-        "#000000",
-        2,
-        this._buildFont(18, true)
-      );
+		let i = areas.findIndex((v) => v.maps.includes(floorId));
+		core.fillRoundRect("ui", 15 - 2, 15 - 2, 35 + 4, 35 + 4, 4, "#444444");
+		core.strokeRoundRect(
+			"ui",
+			15 - 4,
+			15 - 4,
+			35 + 8,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		core.fillBoldText1(
+			"ui",
+			"当前",
+			13 + 20,
+			17 + 20,
+			"#FFFFFF",
+			"#000000",
+			2,
+			this._buildFont(18, true)
+		);
 
-      core.fillRoundRect(
-        "ui",
-        15 + 44 - 2,
-        15 - 2,
-        size + 4 - 58,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 + 44 - 4,
-        15 - 4,
-        size + 8 - 58,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (mapdir.includes("up")) {
-        core.fillBoldText1(
-          "ui",
-          "北▲",
-          30 + 145 + 10,
-          17 + 20,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "北▲",
-          30 + 145 + 10,
-          17 + 20,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
+		core.fillRoundRect(
+			"ui",
+			15 - 2,
+			15 - 2 + 35 + 8 + size + 8 - 54,
+			35 + 4,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4,
+			15 - 4 + 35 + 8 + size + 8 - 54,
+			35 + 8,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (
+			!core.status.maps[core.floorIds[page]].canFlyTo ||
+			!core.hasVisitedFloor(core.floorIds[page])
+		) {
+			core.fillBoldText1(
+				"ui",
+				"预览",
+				13 + 20,
+				17 + 20 + 35 + 8 + size + 8 - 54,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"传送",
+				13 + 20,
+				17 + 20 + 35 + 8 + size + 8 - 54,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
+		core.fillRoundRect(
+			"ui",
+			15 - 4 + size - 2 + 45,
+			15 - 2 + size - 4 + 45,
+			35 + 4,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4 + size - 4 + 45,
+			15 - 4 + size - 4 + 45,
+			35 + 8,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		core.fillBoldText1(
+			"ui",
+			"离开",
+			15 - 4 + size - 4 + 45 + 22,
+			15 - 4 + size - 4 + 45 + 26,
+			"#FFFFFF",
+			"#000000",
+			2,
+			this._buildFont(18, true)
+		);
 
-      core.fillRoundRect(
-        "ui",
-        15 - 2,
-        59 - 2,
-        35 + 4,
-        size + 4 - 58,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4,
-        59 - 4,
-        35 + 8,
-        size + 8 - 58,
-        4,
-        "#444444",
-        1
-      );
-      if (mapdir.includes("left")) {
-        core.fillBoldText1(
-          "ui",
-          "西",
-          15 + 17,
-          25 + 150,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-        core.fillBoldText1(
-          "ui",
-          "◀",
-          15 + 17,
-          45 + 150,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "西",
-          15 + 17,
-          25 + 150,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-        core.fillBoldText1(
-          "ui",
-          "◀",
-          15 + 17,
-          45 + 150,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
-      core.fillRoundRect(
-        "ui",
-        15 + 44 - 2,
-        15 - 2 + size - 4,
-        size + 4 - 58,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 + 44 - 4,
-        15 - 4 + size - 4,
-        size + 8 - 58,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (mapdir.includes("down")) {
-        core.fillBoldText1(
-          "ui",
-          "南▼",
-          30 + 145 + 10,
-          17 + 20 + size - 4,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "南▼",
-          30 + 145 + 10,
-          17 + 20 + size - 4,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
+		core.fillRoundRect(
+			"ui",
+			15 + 44 - 2,
+			15 - 2,
+			size + 4 - 58,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 + 44 - 4,
+			15 - 4,
+			size + 8 - 58,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (mapdir.includes("up")) {
+			core.fillBoldText1(
+				"ui",
+				"北▲",
+				30 + 145 + 10,
+				17 + 20,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"北▲",
+				30 + 145 + 10,
+				17 + 20,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
 
-      core.fillRoundRect(
-        "ui",
-        15 - 2 + size - 4,
-        59 - 2,
-        35 + 4,
-        size + 4 - 58,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4 + size - 4,
-        59 - 4,
-        35 + 8,
-        size + 8 - 58,
-        4,
-        "#444444",
-        1
-      );
-      if (mapdir.includes("right")) {
-        core.fillBoldText1(
-          "ui",
-          "东",
-          15 + 17 + size - 4,
-          25 + 150,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-        core.fillBoldText1(
-          "ui",
-          "▶",
-          15 + 17 + size - 4,
-          45 + 150,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "东",
-          15 + 17 + size - 4,
-          25 + 150,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-        core.fillBoldText1(
-          "ui",
-          "▶",
-          15 + 17 + size - 4,
-          45 + 150,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
-      core.fillRoundRect(
-        "ui",
-        60 - 2,
-        60 - 2,
-        size - 58 + 4,
-        size - 58 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        60 - 4,
-        60 - 4,
-        size - 58 + 8,
-        size - 58 + 8,
-        4,
-        "#444444",
-        1
-      );
-      core.drawThumbnail(floorId, null, {
-        damage: data.damage,
-        ctx: "ui",
-        x: 58,
-        y: 58,
-        size: 0.62,
-        all: data.all,
-      });
-      if (
-        !core.status.maps[core.floorIds[page]].canFlyTo ||
-        !core.hasVisitedFloor(core.floorIds[page])
-      )
-        core.drawImage(
-          "ui",
-          "lock.webp",
-          0,
-          0,
-          size,
-          size,
-          58,
-          58,
-          size - 8,
-          size - 8
-        );
+		core.fillRoundRect(
+			"ui",
+			15 - 2,
+			59 - 2,
+			35 + 4,
+			size + 4 - 58,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4,
+			59 - 4,
+			35 + 8,
+			size + 8 - 58,
+			4,
+			"#444444",
+			1
+		);
+		if (mapdir.includes("left")) {
+			core.fillBoldText1(
+				"ui",
+				"西",
+				15 + 17,
+				25 + 150,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+			core.fillBoldText1(
+				"ui",
+				"◀",
+				15 + 17,
+				45 + 150,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"西",
+				15 + 17,
+				25 + 150,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+			core.fillBoldText1(
+				"ui",
+				"◀",
+				15 + 17,
+				45 + 150,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
+		core.fillRoundRect(
+			"ui",
+			15 + 44 - 2,
+			15 - 2 + size - 4,
+			size + 4 - 58,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 + 44 - 4,
+			15 - 4 + size - 4,
+			size + 8 - 58,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (mapdir.includes("down")) {
+			core.fillBoldText1(
+				"ui",
+				"南▼",
+				30 + 145 + 10,
+				17 + 20 + size - 4,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"南▼",
+				30 + 145 + 10,
+				17 + 20 + size - 4,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
 
-      core.fillRoundRect(
-        "ui",
-        15 + 44 - 2,
-        60 - 2 + size - 4,
-        size + 4 - 58,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 + 44 - 4,
-        60 - 4 + size - 4,
-        size + 8 - 58,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      core.fillBoldText1(
-        "ui",
-        core.status.maps[floorId].areas,
-        30 + 145 + 10,
-        17 + 65 + size - 4,
-        "#FFFFFF",
-        "#000000",
-        2,
-        this._buildFont(18, true)
-      );
-      core.fillRoundRect(
-        "ui",
-        15 - 2,
-        60 - 2 + size - 4,
-        35 + 4,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4,
-        60 - 4 + size - 4,
-        35 + 8,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (i === 0) {
-        core.fillBoldText1(
-          "ui",
-          "◀",
-          30,
-          17 + 65 + size - 4,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "◀",
-          30,
-          17 + 65 + size - 4,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
-      core.fillRoundRect(
-        "ui",
-        15 - 2 + size - 4,
-        60 - 2 + size - 4,
-        35 + 4,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4 + size - 4,
-        60 - 4 + size - 4,
-        35 + 8,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (i === areas.length - 1) {
-        core.fillBoldText1(
-          "ui",
-          "▶",
-          30 + 300 + 10,
-          17 + 65 + size - 4,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "▶",
-          30 + 300 + 10,
-          17 + 65 + size - 4,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
+		core.fillRoundRect(
+			"ui",
+			15 - 2 + size - 4,
+			59 - 2,
+			35 + 4,
+			size + 4 - 58,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4 + size - 4,
+			59 - 4,
+			35 + 8,
+			size + 8 - 58,
+			4,
+			"#444444",
+			1
+		);
+		if (mapdir.includes("right")) {
+			core.fillBoldText1(
+				"ui",
+				"东",
+				15 + 17 + size - 4,
+				25 + 150,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+			core.fillBoldText1(
+				"ui",
+				"▶",
+				15 + 17 + size - 4,
+				45 + 150,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"东",
+				15 + 17 + size - 4,
+				25 + 150,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+			core.fillBoldText1(
+				"ui",
+				"▶",
+				15 + 17 + size - 4,
+				45 + 150,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
+		core.fillRoundRect(
+			"ui",
+			60 - 2,
+			60 - 2,
+			size - 58 + 4,
+			size - 58 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			60 - 4,
+			60 - 4,
+			size - 58 + 8,
+			size - 58 + 8,
+			4,
+			"#444444",
+			1
+		);
+		core.drawThumbnail(floorId, null, {
+			damage: data.damage,
+			ctx: "ui",
+			x: 58,
+			y: 58,
+			size: 0.62,
+			all: data.all,
+		});
+		if (
+			!core.status.maps[core.floorIds[page]].canFlyTo ||
+			!core.hasVisitedFloor(core.floorIds[page])
+		) {
+			/*core.fillRect("ui", 58,
+				58,
+				size - 50,
+				size - 50, "rgba(0,0,0,0.5)")*/
+			core.getContextByName("ui").globalAlpha = 0.7;
 
-      core.fillRoundRect(
-        "ui",
-        15 - 2 + size - 4,
-        15 - 2,
-        80 + 4,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4 + size - 4,
-        15 - 4,
-        80 + 8,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (mapdir.includes("upFloor")) {
-        core.fillBoldText1(
-          "ui",
-          "上楼",
-          30 + 320 + 10,
-          17 + 20,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "上楼",
-          30 + 320 + 10,
-          17 + 20,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
-      core.fillRoundRect(
-        "ui",
-        15 - 2 + size - 4,
-        15 - 2 + size - 4,
-        80 + 4,
-        35 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4 + size - 4,
-        15 - 4 + size - 4,
-        80 + 8,
-        35 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (mapdir.includes("downFloor")) {
-        core.fillBoldText1(
-          "ui",
-          "下楼",
-          30 + 320 + 10,
-          17 + 20 + size - 4,
-          "#FFFFFF",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      } else {
-        core.fillBoldText1(
-          "ui",
-          "下楼",
-          30 + 320 + 10,
-          17 + 20 + size - 4,
-          "#909090",
-          "#000000",
-          2,
-          this._buildFont(18, true)
-        );
-      }
+			core.drawImage(
+				"ui",
+				"miwu.webp",
+				0,
+				0,
+				size,
+				size,
+				58,
+				58,
+				size - 50,
+				size - 50
+			);
+			core.getContextByName("ui").globalAlpha = 1;
+			/*core.fillText("ui", '?', 188,
+				278,
+				"rgba(255,255,255,0.2)", this._buildFont(250, true))*/
+		}
+		core.fillRoundRect(
+			"ui",
+			15 + 44 - 2,
+			60 - 2 + size - 4,
+			size + 4 - 58,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 + 44 - 4,
+			60 - 4 + size - 4,
+			size + 8 - 58,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		core.fillBoldText1(
+			"ui",
+			core.status.maps[floorId].areas,
+			30 + 145 + 10,
+			17 + 65 + size - 4,
+			"#FFFFFF",
+			"#000000",
+			2,
+			this._buildFont(18, true)
+		);
+		core.fillRoundRect(
+			"ui",
+			15 - 2,
+			60 - 2 + size - 4,
+			35 + 4,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4,
+			60 - 4 + size - 4,
+			35 + 8,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (i === 0) {
+			core.fillBoldText1(
+				"ui",
+				"◀",
+				30,
+				17 + 65 + size - 4,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"◀",
+				30,
+				17 + 65 + size - 4,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
+		core.fillRoundRect(
+			"ui",
+			15 - 2 + size - 4,
+			60 - 2 + size - 4,
+			35 + 4,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4 + size - 4,
+			60 - 4 + size - 4,
+			35 + 8,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (i === areas.length - 1) {
+			core.fillBoldText1(
+				"ui",
+				"▶",
+				30 + 300 + 10,
+				17 + 65 + size - 4,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"▶",
+				30 + 300 + 10,
+				17 + 65 + size - 4,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
 
-      core.fillRoundRect(
-        "ui",
-        15 - 2 + size - 4 + 35 + 8,
-        59 - 2,
-        37 + 4,
-        (size - 58) / 2 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4 + size - 4 + 35 + 8,
-        59 - 4,
-        37 + 8,
-        (size - 58) / 2 + 8,
-        4,
-        "#444444",
-        1
-      );
-      const title = core.status.maps[floorId].title;
-      //const length = title.length
-      fillTextVertical(
-        "ui",
-        title,
-        15 - 4 + size - 4 + 45,
-        85,
-        "#FFFFFF",
-        "#000000",
-        18
-      );
-      //const uictx = main.dom.gameCanvas.ui.getContext('2d')
-      core.fillRoundRect(
-        "ui",
-        15 - 2 + size - 4 + 35 + 8,
-        59 - 2 + (size - 58) / 2 + 8,
-        37 + 4,
-        119 + 4,
-        4,
-        "#444444"
-      );
-      core.strokeRoundRect(
-        "ui",
-        15 - 4 + size - 4 + 35 + 8,
-        59 - 4 + (size - 58) / 2 + 8,
-        37 + 8,
-        119 + 8,
-        4,
-        "#444444",
-        1
-      );
-      if (core.getFlag("showEnemy")) {
-        fillTextVertical(
-          "ui",
-          "关闭漏怪检测",
-          15 - 4 + size - 4 + 45,
-          220,
-          "#FFFFFF",
-          "#000000",
-          18
-        );
-      } else {
-        fillTextVertical(
-          "ui",
-          "开启漏怪检测",
-          15 - 4 + size - 4 + 45,
-          220,
-          "#FFFFFF",
-          "#000000",
-          18
-        );
-      }
+		core.fillRoundRect(
+			"ui",
+			15 - 2 + size - 4,
+			15 - 2,
+			80 + 4,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4 + size - 4,
+			15 - 4,
+			80 + 8,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (mapdir.includes("upFloor")) {
+			core.fillBoldText1(
+				"ui",
+				"上楼",
+				30 + 320 + 10,
+				17 + 20,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"上楼",
+				30 + 320 + 10,
+				17 + 20,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
+		core.fillRoundRect(
+			"ui",
+			15 - 2 + size - 4,
+			15 - 2 + size - 4,
+			80 + 4,
+			35 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4 + size - 4,
+			15 - 4 + size - 4,
+			80 + 8,
+			35 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (mapdir.includes("downFloor")) {
+			core.fillBoldText1(
+				"ui",
+				"下楼",
+				30 + 320 + 10,
+				17 + 20 + size - 4,
+				"#FFFFFF",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		} else {
+			core.fillBoldText1(
+				"ui",
+				"下楼",
+				30 + 320 + 10,
+				17 + 20 + size - 4,
+				"#909090",
+				"#000000",
+				2,
+				this._buildFont(18, true)
+			);
+		}
 
-      //uictx.fillTextVertical(title, 15 - 4 + size - 4 + 35 + 29, 25 + 150)
-      //fillTextVertical('ui', title, 15 - 4 + size - 4 + 35 + 29, 25 + 150, '#FFFFFF', this._buildFont(18, true))
-    };
+		core.fillRoundRect(
+			"ui",
+			15 - 2 + size - 4 + 35 + 8,
+			59 - 2,
+			37 + 4,
+			(size - 58) / 2 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4 + size - 4 + 35 + 8,
+			59 - 4,
+			37 + 8,
+			(size - 58) / 2 + 8,
+			4,
+			"#444444",
+			1
+		);
+		const title = core.status.maps[floorId].title;
+		//const length = title.length
+		fillTextVertical(
+			"ui",
+			title,
+			15 - 4 + size - 4 + 45,
+			85,
+			core.hasVisitedFloor(floorId) ? "#FFFFFF" : "#444444",
+			"#000000",
+			18
+		);
+		//const uictx = main.dom.gameCanvas.ui.getContext('2d')
+		core.fillRoundRect(
+			"ui",
+			15 - 2 + size - 4 + 35 + 8,
+			59 - 2 + (size - 58) / 2 + 8,
+			37 + 4,
+			119 + 4,
+			4,
+			"#444444"
+		);
+		core.strokeRoundRect(
+			"ui",
+			15 - 4 + size - 4 + 35 + 8,
+			59 - 4 + (size - 58) / 2 + 8,
+			37 + 8,
+			119 + 8,
+			4,
+			"#444444",
+			1
+		);
+		if (core.getFlag("showEnemy")) {
+			fillTextVertical(
+				"ui",
+				"关闭漏怪检测",
+				15 - 4 + size - 4 + 45,
+				220,
+				"#FFFFFF",
+				"#000000",
+				18
+			);
+		} else {
+			fillTextVertical(
+				"ui",
+				"开启漏怪检测",
+				15 - 4 + size - 4 + 45,
+				220,
+				"#FFFFFF",
+				"#000000",
+				18
+			);
+		}
 
-    function fillTextVertical(name, text, x, y, style, boldstyle, fontsize) {
-      //竖向文字绘制
+		//uictx.fillTextVertical(title, 15 - 4 + size - 4 + 35 + 29, 25 + 150)
+		//fillTextVertical('ui', title, 15 - 4 + size - 4 + 35 + 29, 25 + 150, '#FFFFFF', this._buildFont(18, true))
+	};
 
-      const ctx = core.ui.getContextByName(name);
-      if (!ctx) return;
-      const canvas = document.createElement("canvas");
-      const context = canvas.getContext("2d");
-      fontsize *= 3;
-      const length = text.length;
-      canvas.width = fontsize * 2;
-      canvas.height = fontsize * length * 2;
-      if (style) context.fillStyle = core.arrayToRGBA(style);
-      if (boldstyle) context.strokeStyle = core.arrayToRGBA(boldstyle);
-      context.lineWidth = 2;
-      if (fontsize) context.font = core.ui._buildFont(fontsize, true);
-      let arrText = text.split("");
+	function fillTextVertical(name, text, x, y, style, boldstyle, fontsize) {
+		//竖向文字绘制
 
-      let arrWidth = arrText.map(function (letter) {
-        return context.measureText(letter).width;
-      });
+		const ctx = core.ui.getContextByName(name);
+		if (!ctx) return;
+		const canvas = document.createElement("canvas");
+		const context = canvas.getContext("2d");
+		fontsize *= 3;
+		const length = text.length;
+		canvas.width = fontsize * 2;
+		canvas.height = fontsize * length * 2;
+		if (style) context.fillStyle = core.arrayToRGBA(style);
+		if (boldstyle) context.strokeStyle = core.arrayToRGBA(boldstyle);
+		context.lineWidth = 2;
+		if (fontsize) context.font = core.ui._buildFont(fontsize, true);
+		let arrText = text.split("");
 
-      let align = context.textAlign;
+		let arrWidth = arrText.map(function (letter) {
+			return context.measureText(letter).width;
+		});
 
-      let baseline = context.textBaseline;
-      let sx = fontsize,
-        sy = fontsize * length;
-      if (align == "left") {
-        sx = sx + Math.max.apply(null, arrWidth) / 2;
-      } else if (align == "right") {
-        sx = sx - Math.max.apply(null, arrWidth) / 2;
-      }
+		let align = context.textAlign;
 
-      if (
-        baseline == "bottom" ||
-        baseline == "alphabetic" ||
-        baseline == "ideographic"
-      ) {
-        sy = sy - arrWidth[0] / 2;
-      } else if (baseline == "top" || baseline == "hanging") {
-        sy = sy + arrWidth[0] / 2;
-      }
+		let baseline = context.textBaseline;
+		let sx = fontsize,
+			sy = fontsize * length;
+		if (align == "left") {
+			sx = sx + Math.max.apply(null, arrWidth) / 2;
+		} else if (align == "right") {
+			sx = sx - Math.max.apply(null, arrWidth) / 2;
+		}
 
-      context.textAlign = "center";
+		if (
+			baseline == "bottom" ||
+			baseline == "alphabetic" ||
+			baseline == "ideographic"
+		) {
+			sy = sy - arrWidth[0] / 2;
+		} else if (baseline == "top" || baseline == "hanging") {
+			sy = sy + arrWidth[0] / 2;
+		}
 
-      context.textBaseline = "middle";
-      context.lineWidth = 6;
+		context.textAlign = "center";
 
-      // 开始逐字绘制
+		context.textBaseline = "middle";
+		context.lineWidth = 6;
 
-      arrText.forEach(function (letter, index) {
-        // 确定下一个字符的纵坐标位置
+		// 开始逐字绘制
 
-        context.strokeText(letter, sx, sy);
-        context.fillText(letter, sx, sy);
-        // 旋转坐标系还原成初始态
+		arrText.forEach(function (letter, index) {
+			// 确定下一个字符的纵坐标位置
 
-        context.setTransform(1, 0, 0, 1, 0, 0);
+			context.strokeText(letter, sx, sy);
+			context.fillText(letter, sx, sy);
+			// 旋转坐标系还原成初始态
 
-        // 确定下一个字符的纵坐标位置
+			context.setTransform(1, 0, 0, 1, 0, 0);
 
-        var letterWidth = 54;
+			// 确定下一个字符的纵坐标位置
 
-        sy = sy + letterWidth;
-      });
+			var letterWidth = 54;
 
-      // 水平垂直对齐方式还原
+			sy = sy + letterWidth;
+		});
 
-      context.textAlign = align;
+		// 水平垂直对齐方式还原
 
-      context.textBaseline = baseline;
+		context.textAlign = align;
 
-      //绘制到目标位置
-      ctx.drawImage(
-        canvas,
-        x,
-        y - (fontsize / 3) * length,
-        canvas.width / 3,
-        canvas.height / 3
-      );
-    }
+		context.textBaseline = baseline;
 
-    ////// 点击楼层传送器时的打开操作 //////
-    events.prototype.useFly = function (fromUserAction) {
-      if (core.isReplaying()) return;
-      if (!core.status.maps[core.status.floorId].canFlyFrom) {
-        core.drawTip(core.material.items["fly"].name + "好像失效了", "fly");
-        return;
-      }
-      // 从“浏览地图”页面:尝试直接传送到该层
-      if (core.status.event.id == "viewMaps") {
-        if (!core.hasItem("fly")) {
-          core.playSound("操作失败");
-          core.drawTip("你没有" + core.material.items["fly"].name, "fly");
-        } else if (
-          core.flags.flyNearStair &&
-          !core.nearStair() &&
-          !core.canMoveFloor()
-        ) {
-          core.playSound("操作失败");
-          core.drawTip(
-            "无法到达楼梯边使用" + core.material.items["fly"].name,
-            "fly"
-          );
-        } else {
-          core.flyTo(core.status.event.data.floorId);
-        }
-        return;
-      }
+		//绘制到目标位置
+		ctx.drawImage(
+			canvas,
+			x,
+			y - (fontsize / 3) * length,
+			canvas.width / 3,
+			canvas.height / 3
+		);
+	}
 
-      if (!this._checkStatus("fly", fromUserAction, true)) return;
-      //if (core.flags.flyNearStair && !core.nearStair())
+	////// 点击楼层传送器时的打开操作 //////
+	events.prototype.useFly = function (fromUserAction) {
+		if (!core.isPlaying()) return;
+		if (!core.status.maps[core.status.floorId].canFlyFrom) {
+			core.drawTip(core.material.items["fly"].name + "好像失效了", "fly");
+			return;
+		}
+		// 从“浏览地图”页面:尝试直接传送到该层
+		if (core.status.event.id == "viewMaps") {
+			if (!core.hasItem("fly")) {
+				core.playSound("操作失败");
+				core.drawTip("你没有" + core.material.items["fly"].name, "fly");
+			} else if (
+				core.flags.flyNearStair &&
+				!core.nearStair() &&
+				!flags.canMoveFloor
+			) {
+				core.playSound("操作失败");
+				core.drawTip(
+					"无法到达楼梯边使用" + core.material.items["fly"].name,
+					"fly"
+				);
+			} else {
+				core.flyTo(core.status.event.data.floorId);
+				core.updateStatusBar();
+			}
+			return;
+		}
 
-      if (
-        (core.flags.flyNearStair && !core.nearStair()) ||
-        !core.canMoveFloor()
-      ) {
-        core.playSound("操作失败");
-        core.drawTip(
-          "无法到达楼梯边使用" + core.material.items["fly"].name,
-          "fly"
-        );
-        core.unlockControl();
-        core.status.event.data = null;
-        core.status.event.id = null;
-        return;
-      }
-      if (!core.canUseItem("fly")) {
-        core.playSound("操作失败");
-        core.drawTip(core.material.items["fly"].name + "好像失效了", "fly");
-        core.unlockControl();
-        core.status.event.data = null;
-        core.status.event.id = null;
-        return;
-      }
-      core.playSound("打开界面");
-      core.useItem("fly", true);
-      return;
-    };
-    ////// 系统菜单栏界面时的点击操作 //////
-    actions.prototype._clickSettings = function (x, y) {
-      if (this._out(x)) return;
-      var choices = core.status.event.ui.choices;
-      var topIndex = this._getChoicesTopIndex(choices.length);
-      if (y >= topIndex && y < topIndex + choices.length) {
-        var selection = y - topIndex;
-        core.status.event.selection = selection;
-        switch (selection) {
-          case 0:
-            core.status.event.selection = 0;
-            core.playSound("确定");
-            core.ui._drawSwitchs();
-            break;
-          case 1:
-            // core.playSound('确定');
-            core.ui._drawKeyBoard();
-            break;
-          case 2:
-            // core.playSound('确定');
-            core.clearUI();
-            core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
-            break;
-          case 3:
-            core.status.event.selection = 0;
-            core.playSound("确定");
-            core.ui._drawNotes();
-            break;
-          case 4:
-            core.status.event.selection = 0;
-            core.playSound("确定");
-            core.ui._drawSyncSave();
-            break;
-          case 5:
-            core.status.event.selection = 0;
-            core.playSound("确定");
-            core.ui._drawGameInfo();
-            break;
-          case 6:
-            return core.confirmRestart();
-          case 7:
-            core.playSound("取消");
-            core.ui.closePanel();
-            break;
-        }
-      }
-      return;
-    };
-    ////// 查看地图界面时的点击操作 //////
-    actions.prototype._clickViewMaps = function (x, y, px, py) {
-      if (core.status.event.data == null) {
-        core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
-        return;
-      }
-      let now = core.floorIds.indexOf(core.status.floorId);
-      let index = core.status.event.data.index;
-      let cx = core.status.event.data.x,
-        cy = core.status.event.data.y;
-      let floorId = core.floorIds[index],
-        mw = core.floors[floorId].width,
-        mh = core.floors[floorId].height;
-      let perpx = core.__PIXELS__ / 5,
-        cornerpx = (perpx * 3) / 4;
-      const bfs = core.plugin.bfsSearch(floorId, 1, true);
-      const mapdir = bfs.mapdir[floorId];
-      const res = bfs.res;
-      const formto = {};
-      for (let from in res) {
-        const to = res[from];
-        const [fromfloorId, fromsx, fromsy, dir] = from.split("_");
-        const [tofloorId, tosx, tosy] = to.split("_");
-        if (!formto[fromfloorId]) formto[fromfloorId] = {};
-        if (!formto[fromfloorId][dir]) formto[fromfloorId][dir] = tofloorId;
-      }
-      const areas = core.getFlag("areas");
-      let i = areas.findIndex((v) => v.maps.includes(floorId));
+		if (!this._checkStatus("fly", fromUserAction, true)) return;
+		//if (core.flags.flyNearStair && !core.nearStair())
 
-      if (px >= 11 && px <= 54 && py >= 11 && py <= 54) {
-        core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
-      } else if (px >= 362 && px <= 407 && py >= 191 && py <= 318) {
-        flags.showEnemy = !flags.showEnemy;
-        core.ui._drawViewMaps(index);
-      } else if (px >= 364 && px <= 407 && py >= 364 && py <= 407) {
-        core.clearMap("data");
-        core.playSound("取消");
-        core.ui.closePanel();
-        core.getItemDetail();
-        core.redrawMap();
-        core.ui.statusBar._update_map();
-        return;
-      } else if (px >= 55 && px <= 317 && py >= 11 && py <= 54) {
-        if (mapdir.includes("up"))
-          core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].up));
-      } else if (px >= 55 && px <= 317 && py >= 319 && py <= 362) {
-        if (mapdir.includes("down"))
-          core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].down));
-      } else if (px >= 11 && px <= 54 && py >= 55 && py <= 317) {
-        if (mapdir.includes("left"))
-          core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].left));
-      } else if (px >= 319 && px <= 362 && py >= 55 && py <= 317) {
-        if (mapdir.includes("right"))
-          core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].right));
-      } else if (px >= 319 && px <= 407 && py >= 11 && py <= 54) {
-        if (mapdir.includes("upFloor"))
-          core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].upFloor));
-      } else if (px >= 319 && px <= 407 && py >= 319 && py <= 362) {
-        if (mapdir.includes("downFloor"))
-          core.ui._drawViewMaps(
-            core.floorIds.indexOf(formto[floorId].downFloor)
-          );
-      } else if (
-        px >= 55 &&
-        px <= 317 &&
-        py >= 55 &&
-        py <= 317 &&
-        !core.isReplaying()
-      ) {
-        core.useFly(false);
-        return;
-      } else if (px >= 11 && px <= 54 && py >= 364 && py <= 407) {
-        if (i > 0) {
-          i -= 1;
-          core.ui._drawViewMaps(core.floorIds.indexOf(areas[i].maps[0]));
-        }
-      } else if (px >= 319 && px <= 362 && py >= 364 && py <= 407) {
-        if (i < areas.length - 1) {
-          i += 1;
-          core.ui._drawViewMaps(core.floorIds.indexOf(areas[i].maps[0]));
-        }
-      }
-    };
-    const replayAction_fly = function (action) {
-      //楼层传送的录像操作
-      if (action.indexOf("fly:") != 0) return false;
-      var floorId = action.substring(4);
-      var toIndex = core.floorIds.indexOf(floorId);
-      if (
-        !core.canUseItem("fly") ||
-        (core.flags.flyNearStair && !core.nearStair())
-      )
-        return false;
-      core.ui._drawViewMaps(toIndex);
-      if (core.status.replay.speed == 24) {
-        if (!core.flyTo(floorId, core.replay))
-          core.control._replay_error(action);
-        return true;
-      }
-      setTimeout(function () {
-        if (!core.flyTo(floorId, core.replay))
-          core.control._replay_error(action);
-      }, core.control.__replay_getTimeout());
-      return true;
-    };
-    core.registerReplayAction("fly", replayAction_fly);
-    ////// 查看地图界面时,放开某个键的操作 //////
-    actions.prototype._keyUpViewMaps = function (keycode) {
-      if (core.status.event.data == null) {
-        core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
-        return;
-      }
-      var floorId = core.floorIds[core.status.event.data.index];
+		if (
+			(core.flags.flyNearStair && !core.nearStair()) ||
+			!flags.canMoveFloor
+		) {
+			core.playSound("操作失败");
+			core.drawTip(
+				"无法到达楼梯边使用" + core.material.items["fly"].name,
+				"fly"
+			);
+			core.unlockControl();
+			core.status.event.data = null;
+			core.status.event.id = null;
+			return;
+		}
+		if (!core.canUseItem("fly")) {
+			core.playSound("操作失败");
+			core.drawTip(core.material.items["fly"].name + "好像失效了", "fly");
+			core.unlockControl();
+			core.status.event.data = null;
+			core.status.event.id = null;
+			return;
+		}
+		core.playSound("打开界面");
+		core.useItem("fly", true);
+		core.updateStatusBar();
+		return;
+	};
+	////// 系统菜单栏界面时的点击操作 //////
+	actions.prototype._clickSettings = function (x, y) {
+		if (this._out(x)) return;
+		var choices = core.status.event.ui.choices;
+		var topIndex = this._getChoicesTopIndex(choices.length);
+		if (y >= topIndex && y < topIndex + choices.length) {
+			var selection = y - topIndex;
+			core.status.event.selection = selection;
+			switch (selection) {
+			case 0:
+				core.status.event.selection = 0;
+				core.playSound("确定");
+				core.ui._drawSwitchs();
+				break;
+			case 1:
+				// core.playSound('确定');
+				core.ui._drawKeyBoard();
+				break;
+			case 2:
+				// core.playSound('确定');
+				core.clearUI();
+				core.useItem("fly");
+				break;
+			case 3:
+				core.status.event.selection = 0;
+				core.playSound("确定");
+				core.ui._drawNotes();
+				break;
+			case 4:
+				core.status.event.selection = 0;
+				core.playSound("确定");
+				core.ui._drawSyncSave();
+				break;
+			case 5:
+				core.status.event.selection = 0;
+				core.playSound("确定");
+				core.ui._drawGameInfo();
+				break;
+			case 6:
+				return core.confirmRestart();
+			case 7:
+				core.playSound("取消");
+				core.ui.closePanel();
+				break;
+			}
+		}
+		return;
+	};
+	////// 查看地图界面时的点击操作 //////
+	actions.prototype._clickViewMaps = function (x, y, px, py) {
+		if (core.status.event.data == null) {
+			core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
+			return;
+		}
+		let now = core.floorIds.indexOf(core.status.floorId);
+		let index = core.status.event.data.index;
+		let cx = core.status.event.data.x,
+			cy = core.status.event.data.y;
+		let floorId = core.floorIds[index],
+			mw = core.floors[floorId].width,
+			mh = core.floors[floorId].height;
+		let perpx = core.__PIXELS__ / 5,
+			cornerpx = (perpx * 3) / 4;
+		const bfs = core.plugin.bfsSearch(floorId, 1, true);
+		const mapdir = bfs.mapdir[floorId];
+		const res = bfs.res;
+		const formto = {};
+		for (let from in res) {
+			const to = res[from];
+			const [fromfloorId, fromsx, fromsy, dir] = from.split("_");
+			const [tofloorId, tosx, tosy] = to.split("_");
+			if (!formto[fromfloorId]) formto[fromfloorId] = {};
+			if (!formto[fromfloorId][dir]) formto[fromfloorId][dir] = tofloorId;
+		}
+		const areas = core.getFlag("areas");
+		let i = areas.findIndex((v) => v.maps.includes(floorId));
 
-      if (keycode == 27 || keycode == 71) {
-        core.clearMap("data");
-        core.playSound("取消");
-        core.ui.closePanel();
-        core.getItemDetail();
-        core.redrawMap();
-        core.ui.statusBar._update_map();
-        return;
-      }
-      if (keycode == 88) {
-        core.openBook(true);
-        return;
-      }
-      if (keycode == 86) {
-        core.status.event.data.damage = !core.status.event.data.damage;
-        core.playSound("光标移动");
-        core.ui._drawViewMaps(core.status.event.data);
-        return;
-      }
-      if (keycode == 66) {
-        core.openBook(false);
-        return;
-      }
-      if (
-        (keycode == 13 || keycode == 32 || keycode == 67) &&
-        !core.isReplaying()
-      ) {
-        core.useFly(false);
-        return;
-      }
-      return;
-    };
-    actions.prototype._keyDownViewMaps = function (keycode) {
-      if (core.status.event.data == null) return;
+		if (px >= 11 && px <= 54 && py >= 11 && py <= 54) {
+			core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
+		} else if (px >= 362 && px <= 407 && py >= 191 && py <= 318) {
+			flags.showEnemy = !flags.showEnemy;
+			core.ui._drawViewMaps(index);
+		} else if (px >= 364 && px <= 407 && py >= 364 && py <= 407) {
+			core.clearMap("data");
+			core.playSound("取消");
+			core.ui.closePanel();
+			core.getItemDetail();
+			core.redrawMap();
+			core.updateStatusBar();
+			core.ui.statusBar._update_map();
+			return;
+		} else if (px >= 55 && px <= 317 && py >= 11 && py <= 54) {
+			if (mapdir.includes("up"))
+				core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].up));
+		} else if (px >= 55 && px <= 317 && py >= 319 && py <= 362) {
+			if (mapdir.includes("down"))
+				core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].down));
+		} else if (px >= 11 && px <= 54 && py >= 55 && py <= 317) {
+			if (mapdir.includes("left"))
+				core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].left));
+		} else if (px >= 319 && px <= 362 && py >= 55 && py <= 317) {
+			if (mapdir.includes("right"))
+				core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].right));
+		} else if (px >= 319 && px <= 407 && py >= 11 && py <= 54) {
+			if (mapdir.includes("upFloor"))
+				core.ui._drawViewMaps(core.floorIds.indexOf(formto[floorId].upFloor));
+		} else if (px >= 319 && px <= 407 && py >= 319 && py <= 362) {
+			if (mapdir.includes("downFloor"))
+				core.ui._drawViewMaps(
+					core.floorIds.indexOf(formto[floorId].downFloor)
+				);
+		} else if (
+			px >= 55 &&
+			px <= 317 &&
+			py >= 55 &&
+			py <= 317 &&
+			core.isPlaying()
+		) {
+			core.useFly(false);
+			return;
+		} else if (px >= 11 && px <= 54 && py >= 364 && py <= 407) {
+			if (i > 0) {
+				i -= 1;
+				core.ui._drawViewMaps(core.floorIds.indexOf(areas[i].maps[0]));
+			}
+		} else if (px >= 319 && px <= 362 && py >= 364 && py <= 407) {
+			if (i < areas.length - 1) {
+				i += 1;
+				core.ui._drawViewMaps(core.floorIds.indexOf(areas[i].maps[0]));
+			}
+		}
+	};
+	const replayAction_fly = function (action) {
+		//楼层传送的录像操作
+		if (action.indexOf("fly:") != 0) return false;
+		var floorId = action.substring(4);
+		var toIndex = core.floorIds.indexOf(floorId);
+		if (
+			!core.canUseItem("fly") ||
+			(core.flags.flyNearStair && !core.nearStair() && !flags.canMoveFloor)
+		)
+			return false;
+		core.ui._drawViewMaps(toIndex);
+		if (core.status.replay.speed == 24) {
+			if (!core.flyTo(floorId, core.replay))
+				core.control._replay_error(action);
+			return true;
+		}
+		setTimeout(function () {
+			if (!core.flyTo(floorId, core.replay))
+				core.control._replay_error(action);
+		}, core.control.__replay_getTimeout());
+		return true;
+	};
+	core.registerReplayAction("fly", replayAction_fly);
+	////// 查看地图界面时,放开某个键的操作 //////
 
-      var floorId = core.floorIds[core.status.event.data.index],
-        mh = core.floors[floorId].height;
+	actions.prototype._keyUpViewMaps = function (keycode) {
+		if (core.status.event.data == null) {
+			core.ui._drawViewMaps(core.floorIds.indexOf(core.status.floorId));
+			return;
+		}
+		var floorId = core.floorIds[core.status.event.data.index];
 
-      if (keycode == 39) this._clickViewMaps(9, 1, 330, 250);
-      if (keycode == 37) this._clickViewMaps(9, 8, 25, 200);
-      if (keycode == 40) this._clickViewMaps(9, 6, 250, 330);
-      if (keycode == 38) this._clickViewMaps(9, 3, 200, 25);
-      if (keycode == 34) this._clickViewMaps(9, 3, 350, 330);
-      if (keycode == 33) this._clickViewMaps(9, 3, 350, 25);
-      return;
-    };
+		if (keycode == 27 || keycode == 71) {
+			core.clearMap("data");
+			core.playSound("取消");
+			core.ui.closePanel();
+			core.getItemDetail();
+			core.redrawMap();
+			core.ui.statusBar._update_map();
+			core.updateStatusBar();
+			return;
+		}
 
-    actions.prototype._sys_onmousewheel = function (direct) {
-      // 向下滚动是 -1 ,向上是 1
+		if (keycode == 86) {
+			core.status.event.data.damage = !core.status.event.data.damage;
+			core.playSound("光标移动");
+			core.ui._drawViewMaps(core.status.event.data);
+			return;
+		}
+		if (keycode == 66 || keycode == 88) {
+			if (core.isReplaying()) {
+				core.control._replay_book();
+			} else {
+				core.openBook(false);
+			}
+			return;
+		}
+		if (
+			(keycode == 13 || keycode == 32 || keycode == 67) &&
+			!core.isReplaying()
+		) {
+			core.useFly(false);
+			return;
+		}
+		return;
+	};
+	actions.prototype._keyDownViewMaps = function (keycode) {
+		if (core.status.event.data == null) return;
 
-      if (this._checkReplaying()) {
-        // 滚轮控制速度
-        if (direct == 1) core.speedUpReplay();
-        if (direct == -1) core.speedDownReplay();
-        return;
-      }
+		var floorId = core.floorIds[core.status.event.data.index],
+			mh = core.floors[floorId].height;
 
-      // 楼层飞行器
-      if (core.status.lockControl && core.status.event.id == "fly") {
-        if (direct == 1) core.ui.drawFly(this._getNextFlyFloor(1));
-        if (direct == -1) core.ui.drawFly(this._getNextFlyFloor(-1));
-        return;
-      }
+		if (keycode == 39) this._clickViewMaps(9, 1, 330, 250);
+		if (keycode == 37) this._clickViewMaps(9, 8, 25, 200);
+		if (keycode == 40) this._clickViewMaps(9, 6, 250, 330);
+		if (keycode == 38) this._clickViewMaps(9, 3, 200, 25);
+		if (keycode == 34) this._clickViewMaps(9, 3, 350, 330);
+		if (keycode == 33) this._clickViewMaps(9, 3, 350, 25);
+		return;
+	};
 
-      // 怪物手册
-      if (core.status.lockControl && core.status.event.id == "book") {
-        var pageinfo = core.ui._drawBook_pageinfo();
-        if (direct == 1)
-          core.ui.drawBook(core.status.event.data - pageinfo.per_page);
-        if (direct == -1)
-          core.ui.drawBook(core.status.event.data + pageinfo.per_page);
-        return;
-      }
+	actions.prototype._sys_onmousewheel = function (direct) {
+		// 向下滚动是 -1 ,向上是 1
 
-      // 存读档
-      if (
-        core.status.lockControl &&
-        (core.status.event.id == "save" || core.status.event.id == "load")
-      ) {
-        var index =
-          core.status.event.data.page * 10 + core.status.event.data.offset;
-        if (direct == 1) core.ui._drawSLPanel(index - 10);
-        if (direct == -1) core.ui._drawSLPanel(index + 10);
-        return;
-      }
+		if (this._checkReplaying()) {
+			// 滚轮控制速度
+			if (direct == 1) core.speedUpReplay();
+			if (direct == -1) core.speedDownReplay();
+			return;
+		}
 
-      // 浏览地图
-      if (core.status.lockControl && core.status.event.id == "viewMaps") {
-        let floorId = core.floorIds[core.status.event.data.index];
-        if (!flags.__visited__[floorId]) floorId = core.status.floorId;
-        const visit = Object.keys(flags.__visited__);
-        let index = visit.indexOf(floorId);
-        if (direct == 1) {
-          if (index > 0)
-            core.ui._drawViewMaps(core.floorIds.indexOf(visit[index - 1]));
-        }
-        if (direct == -1) {
-          if (index < visit.length - 1)
-            core.ui._drawViewMaps(core.floorIds.indexOf(visit[index + 1]));
-        }
-        return;
-      }
+		// 楼层飞行器
+		if (core.status.lockControl && core.status.event.id == "fly") {
+			if (direct == 1) core.ui.drawFly(this._getNextFlyFloor(1));
+			if (direct == -1) core.ui.drawFly(this._getNextFlyFloor(-1));
+			return;
+		}
 
-      // wait事件
-      if (
-        core.status.lockControl &&
-        core.status.event.id == "action" &&
-        core.status.event.data.type == "wait"
-      ) {
-        var timeout =
-          Math.max(0, core.status.event.timeout - new Date().getTime()) || 0;
-        core.setFlag("type", 0);
-        var keycode = direct == 1 ? 33 : 34;
-        core.setFlag("keycode", keycode);
-        core.setFlag("timeout", timeout);
-        var executed = core.events.__action_wait_afterGet(
-          core.status.event.data.current
-        );
-        if (executed || !core.status.event.data.current.forceChild) {
-          core.status.route.push("input:" + (1e8 * timeout + keycode));
-          clearTimeout(core.status.event.interval);
-          delete core.status.event.timeout;
-          core.doAction();
-        }
-        return;
-      }
-    };
-    core.registerAction(
-      "onmousewheel",
-      "_sys_onmousewheel",
-      actions.prototype._sys_onmousewheel,
-      0
-    );
-  },
+		// 怪物手册
+		if (core.status.lockControl && core.status.event.id == "book") {
+			var pageinfo = core.ui._drawBook_pageinfo();
+			if (direct == 1)
+				core.ui.drawBook(core.status.event.data - pageinfo.per_page);
+			if (direct == -1)
+				core.ui.drawBook(core.status.event.data + pageinfo.per_page);
+			return;
+		}
+
+		// 存读档
+		if (
+			core.status.lockControl &&
+			(core.status.event.id == "save" || core.status.event.id == "load")
+		) {
+			var index =
+				core.status.event.data.page * 10 + core.status.event.data.offset;
+			if (direct == 1) core.ui._drawSLPanel(index - 10);
+			if (direct == -1) core.ui._drawSLPanel(index + 10);
+			return;
+		}
+
+		// 浏览地图
+		if (core.status.lockControl && core.status.event.id == "viewMaps") {
+			let floorId = core.floorIds[core.status.event.data.index];
+			if (!flags.__visited__[floorId]) floorId = core.status.floorId;
+			const visit = Object.keys(flags.__visited__);
+			let index = visit.indexOf(floorId);
+			if (direct == 1) {
+				if (index > 0)
+					core.ui._drawViewMaps(core.floorIds.indexOf(visit[index - 1]));
+			}
+			if (direct == -1) {
+				if (index < visit.length - 1)
+					core.ui._drawViewMaps(core.floorIds.indexOf(visit[index + 1]));
+			}
+			return;
+		}
+
+		// wait事件
+		if (
+			core.status.lockControl &&
+			core.status.event.id == "action" &&
+			core.status.event.data.type == "wait"
+		) {
+			var timeout =
+				Math.max(0, core.status.event.timeout - new Date().getTime()) || 0;
+			core.setFlag("type", 0);
+			var keycode = direct == 1 ? 33 : 34;
+			core.setFlag("keycode", keycode);
+			core.setFlag("timeout", timeout);
+			var executed = core.events.__action_wait_afterGet(
+				core.status.event.data.current
+			);
+			if (executed || !core.status.event.data.current.forceChild) {
+				core.status.route.push("input:" + (1e8 * timeout + keycode));
+				clearTimeout(core.status.event.interval);
+				delete core.status.event.timeout;
+				core.doAction();
+			}
+			return;
+		}
+	};
+	core.registerAction(
+		"onmousewheel",
+		"_sys_onmousewheel",
+		actions.prototype._sys_onmousewheel,
+		0
+	);
+},
     "CG回廊": function () {
-    // 在此增加新插件
-    const CGUI = document.createElement("canvas"); //CGui画布设置
-    CGUI.style.position = "absolute";
-    CGUI.style.zIndex = 300;
-    CGUI.style.display = "none";
-    CGUI.id = "CGUI";
-    main.dom.gameGroup.insertAdjacentElement("afterend", CGUI);
-    CGUI.style.top = "50%";
-    CGUI.style.left = "50%";
-    CGUI.style.transform = "translate(-50%,-50%)";
-    const ctx = CGUI.getContext("2d");
-    main.dom.CGUI = CGUI;
-    let page = 0; //初始页面
-    let show = false; //展示状态
-    CGUI.onclick = function (e) {
-      try {
-        e.preventDefault();
-        if (core.isPlaying()) return false;
-        const left = core.dom.gameGroup.offsetLeft;
-        const top = core.dom.gameGroup.offsetTop;
-        const px = Math.floor((e.clientX - left) / core.domStyle.scale),
-          py = Math.floor((e.clientY - top) / core.domStyle.scale);
-        core.ui.CG.onclick(px * 3, py * 3);
-      } catch (ee) {
-        main.log(ee);
-      }
-    };
+	// 在此增加新插件
+	const CGUI = document.createElement("canvas"); //CGui画布设置
+	CGUI.style.position = "absolute";
+	CGUI.style.zIndex = 300;
+	CGUI.style.display = "none";
+	CGUI.id = "CGUI";
+	main.dom.gameGroup.insertAdjacentElement("afterend", CGUI);
+	CGUI.style.top = "50%";
+	CGUI.style.left = "50%";
+	CGUI.style.transform = "translate(-50%,-50%)";
+	const ctx = CGUI.getContext("2d");
+	main.dom.CGUI = CGUI;
+	let page = 0; //初始页面
+	let show = false; //展示状态
+	CGUI.onclick = function (e) {
+		try {
+			e.preventDefault();
+			if (core.isPlaying()) return false;
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+				py = Math.floor((e.clientY - top) / core.domStyle.scale);
+			core.ui.CG.onclick(px * 3, py * 3);
+		} catch (ee) {
+			main.log(ee);
+		}
+	};
 
-    class CG {
-      constructor() {
-        this.cgs;
-        //cg列表
-        this.UIMx = [
-          //空位用‘none’填充,当前ui至多4列6行
-          [
-            ["eve_010102.webp", "eve_010203.webp", "eve_010304.webp"],
-            ["eve_010501.webp", "eve_010601.webp", "eve_010701.webp"],
-          ],
-          [
-            ["eve_010801.webp", "eve_010902.webp", "eve_011001.webp"],
-            ["eve_011101.webp", "eve_011202.webp", "eve_011302.webp"],
-          ],
-          [
-            ["eve_011402.webp", "eve_020102.webp", "eve_020201.webp"],
-            ["eve_020301.webp", "eve_020401.webp", "eve_020501.webp"],
-          ],
-          [
-            ["eve_020605.webp", "eve_020701.webp", "eve_020801.webp"],
-            ["eve_030101.webp", "eve_030206.webp", "eve_030302.webp"],
-          ],
-          [
-            ["eve_030508.webp", "eve_030601.webp", "eve_030801.webp"],
-            ["eve_030901.webp", "eve_031002.webp", "eve_031101.webp"],
-          ],
-          [
-            ["eve_040201.webp", "eve_040401.webp", "eve_040501.webp"],
-            ["eve_040601.webp", "eve_040702.webp", "eve_040801.webp"],
-          ],
-          [
-            ["eve_050101.webp", "eve_050201.webp", "eve_050401.webp"],
-            ["eve_050501.webp", "eve_050601.webp", "eve_050704.webp"],
-          ],
-          [
-            ["eve_050801.webp", "eve_070101.webp", "bg_1511.webp"],
-            ["bg_1521.webp", "bg_2011.webp", "bg_2521.webp"],
-          ],
-          [
-            ["bg_3042.webp", "bg_3551.webp", "bg_3571.webp"],
-            ["bg_3721.webp", "bg_5033.webp", "bg_5044.webp"],
-          ],
-        ];
-      }
+	class CG {
+		constructor() {
+			this.cgs;
+			//cg列表
+			this.UIMx = [
+				//空位用‘none’填充,当前ui3*2
+				[
+					["eve_010102.webp", "eve_010203.webp", "eve_010304.webp"],
+					["eve_010501.webp", "eve_010601.webp", "eve_010701.webp"],
+				],
+				[
+					["eve_010801.webp", "eve_010902.webp", "eve_011001.webp"],
+					["eve_011101.webp", "eve_011202.webp", "eve_011302.webp"],
+				],
+				[
+					["eve_011402.webp", "eve_020102.webp", "eve_020201.webp"],
+					["eve_020301.webp", "eve_020401.webp", "eve_020501.webp"],
+				],
+				[
+					["eve_020605.webp", "eve_020701.webp", "eve_020801.webp"],
+					["eve_030101.webp", "eve_030206.webp", "eve_030302.webp"],
+				],
+				[
+					["eve_030508.webp", "eve_030601.webp", "eve_030801.webp"],
+					["eve_030901.webp", "eve_031002.webp", "eve_031101.webp"],
+				],
+				[
+					["eve_040201.webp", "eve_040401.webp", "eve_040501.webp"],
+					["eve_040601.webp", "eve_040702.webp", "eve_040801.webp"],
+				],
+				[
+					["eve_050101.webp", "eve_050201.webp", "eve_050401.webp"],
+					["eve_050501.webp", "eve_050601.webp", "eve_050704.webp"],
+				],
+				[
+					["eve_050801.webp", "eve_070101.webp", "bg_1511.webp"],
+					["bg_1521.webp", "bg_2011.webp", "bg_2521.webp"],
+				],
+				[
+					["bg_3042.webp", "bg_3551.webp", "bg_3571.webp"],
+					["bg_3721.webp", "bg_5033.webp", "bg_5044.webp"],
+				],
+			];
+		}
 
-      //更新
-      update() {
-        this.background();
-        this.drawUI();
-      }
-      background() {
-        //画布大小设置
-        if (core.domStyle.isVertical) {
-          CGUI.width = 1248;
-          CGUI.height = 2028;
-        } else {
-          CGUI.width = 2028;
-          CGUI.height = 1248;
-        }
-        core.setTextAlign(ctx, "center");
-      }
-      onclick(px, py) {
-        //点击
+		//更新
+		update() {
+			this.background();
+			this.drawUI();
+		}
+		background() {
+			//画布大小设置
+			if (core.domStyle.isVertical) {
+				CGUI.width = 1248;
+				CGUI.height = 2028;
+			} else {
+				CGUI.width = 2028;
+				CGUI.height = 1248;
+			}
+			core.setTextAlign(ctx, "center");
+		}
+		onclick(px, py) {
+			//点击
 
-        if (show) {
-          show = !show;
-          core.clearMap(ctx);
-          this.update();
-          return;
-        }
-        const makeBox = ([x, y], [w, h]) => {
-          return [
-            [x, y],
-            [x + w, y + h],
-          ];
-        };
-        const inRect = ([x, y], [[sx, sy], [dx, dy]]) => {
-          return sx <= x && x <= dx && sy <= y && y <= dy;
-        };
-        const pos = [px, py];
-        const backbox = makeBox([15, 35], [210, 90]);
-        if (inRect(pos, backbox)) {
-          //离开按钮是一致的,其余的记区分横竖屏
-          CGUI.style.display = "none";
-          core.clearMap(ctx);
-          core.restart();
-          return;
-        }
-        if (core.domStyle.isVertical) {
-          //竖屏
-          const pageupbox = makeBox([200, 1830], [200, 100]);
-          const pagedownbox = makeBox([900, 1830], [200, 100]);
+			if (show) {
+				show = !show;
+				core.clearMap(ctx);
+				this.update();
+				return;
+			}
+			const makeBox = ([x, y], [w, h]) => {
+				return [
+					[x, y],
+					[x + w, y + h],
+				];
+			};
+			const inRect = ([x, y], [
+				[sx, sy],
+				[dx, dy]
+			]) => {
+				return sx <= x && x <= dx && sy <= y && y <= dy;
+			};
+			const pos = [px, py];
+			const backbox = makeBox([15, 35], [210, 90]);
+			if (inRect(pos, backbox)) {
+				//离开按钮是一致的,其余的记区分横竖屏
+				CGUI.style.display = "none";
+				core.clearMap(ctx);
+				core.restart();
+				return;
+			}
+			if (core.domStyle.isVertical) {
+				//竖屏
+				const pageupbox = makeBox([200, 1830], [200, 100]);
+				const pagedownbox = makeBox([900, 1830], [200, 100]);
 
-          const imagebox0 = makeBox([50, 200], [560, 420]);
-          const imagebox1 = makeBox([50, 750], [560, 420]);
-          const imagebox2 = makeBox([50, 1300], [560, 420]);
+				const imagebox0 = makeBox([50, 200], [560, 420]);
+				const imagebox1 = makeBox([50, 750], [560, 420]);
+				const imagebox2 = makeBox([50, 1300], [560, 420]);
 
-          const imagebox3 = makeBox([650, 200], [560, 420]);
-          const imagebox4 = makeBox([650, 750], [560, 420]);
-          const imagebox5 = makeBox([650, 1300], [560, 420]);
-          if (inRect(pos, pagedownbox)) {
-            //2代表当前最大页数-1
-            if (page < this.UIMx.length - 1) {
-              page++;
-              core.clearMap(ctx);
-              this.update();
-            }
-          } else if (inRect(pos, pageupbox)) {
-            if (page > 0) {
-              page--;
-              core.clearMap(ctx);
-              this.update();
-            }
-          } else if (inRect(pos, imagebox0)) {
-            if (this.cgs.includes(this.UIMx[page][0][0])) {
-              const img = core.material.images.images[this.UIMx[page][0][0]];
-              if (img) {
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                ctx.restore(); //重置画布设置
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox1)) {
-            if (this.cgs.includes(this.UIMx[page][0][1])) {
-              const img = core.material.images.images[this.UIMx[page][0][1]];
-              if (img) {
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                ctx.restore(); //重置画布设置
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox2)) {
-            if (this.cgs.includes(this.UIMx[page][0][2])) {
-              const img = core.material.images.images[this.UIMx[page][0][2]];
-              if (img) {
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                ctx.restore(); //重置画布设置
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox3)) {
-            if (this.cgs.includes(this.UIMx[page][1][0])) {
-              const img = core.material.images.images[this.UIMx[page][1][0]];
-              if (img) {
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                ctx.restore(); //重置画布设置
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox4)) {
-            if (this.cgs.includes(this.UIMx[page][1][1])) {
-              const img = core.material.images.images[this.UIMx[page][1][1]];
-              if (img) {
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                ctx.restore(); //重置画布设置
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox5)) {
-            if (this.cgs.includes(this.UIMx[page][1][2])) {
-              const img = core.material.images.images[this.UIMx[page][1][2]];
-              if (img) {
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                ctx.restore(); //重置画布设置
-                show = !show;
-              }
-            }
-          }
-        } else {
-          const pageupbox = makeBox([200, 1110], [200, 100]);
-          const pagedownbox = makeBox([1600, 1110], [200, 100]);
-          const imagebox0 = makeBox([75, 150], [600, 450]);
-          const imagebox1 = makeBox([725, 150], [600, 450]);
-          const imagebox2 = makeBox([1300, 150], [600, 450]);
-          const imagebox3 = makeBox([75, 650], [600, 450]);
-          const imagebox4 = makeBox([725, 650], [600, 450]);
-          const imagebox5 = makeBox([1375, 650], [600, 450]);
-          if (inRect(pos, pagedownbox)) {
-            if (page < this.UIMx.length - 1) {
-              page++;
-              core.clearMap(ctx);
-              this.update();
-            }
-          } else if (inRect(pos, pageupbox)) {
-            if (page > 0) {
-              page--;
-              core.clearMap(ctx);
-              this.update();
-            }
-          } else if (inRect(pos, imagebox0)) {
-            if (this.cgs.includes(this.UIMx[page][0][0])) {
-              const img = core.material.images.images[this.UIMx[page][0][0]];
-              if (img) {
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox1)) {
-            if (this.cgs.includes(this.UIMx[page][0][1])) {
-              const img = core.material.images.images[this.UIMx[page][0][1]];
-              if (img) {
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox2)) {
-            if (this.cgs.includes(this.UIMx[page][0][2])) {
-              const img = core.material.images.images[this.UIMx[page][0][2]];
-              if (img) {
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox3)) {
-            if (this.cgs.includes(this.UIMx[page][1][0])) {
-              const img = core.material.images.images[this.UIMx[page][1][0]];
-              if (img) {
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox4)) {
-            if (this.cgs.includes(this.UIMx[page][1][1])) {
-              const img = core.material.images.images[this.UIMx[page][1][1]];
-              if (img) {
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                show = !show;
-              }
-            }
-          } else if (inRect(pos, imagebox5)) {
-            if (this.cgs.includes(this.UIMx[page][1][2])) {
-              const img = core.material.images.images[this.UIMx[page][1][2]];
-              if (img) {
-                ctx.drawImage(img, 0, 0, 2028, 1248);
-                show = !show;
-              }
-            }
-          }
-        }
-      }
-      drawUI() {
-        //绘制页面
-        core.clearMap(CGUI);
-        const bgVertical = core.material.images.images["bg_2010.webp"]; //竖屏背景
-        const bg = core.material.images.images["bg_5043.webp"]; //横屏背景
+				const imagebox3 = makeBox([650, 200], [560, 420]);
+				const imagebox4 = makeBox([650, 750], [560, 420]);
+				const imagebox5 = makeBox([650, 1300], [560, 420]);
+				if (inRect(pos, pagedownbox)) {
+					//2代表当前最大页数-1
+					if (page < this.UIMx.length - 1) {
+						page++;
+						core.clearMap(ctx);
+						this.update();
+					}
+				} else if (inRect(pos, pageupbox)) {
+					if (page > 0) {
+						page--;
+						core.clearMap(ctx);
+						this.update();
+					}
+				} else if (inRect(pos, imagebox0)) {
+					if (this.cgs.includes(this.UIMx[page][0][0])) {
+						const img = core.material.images.images[this.UIMx[page][0][0]];
+						if (img) {
+							ctx.save(); //保存设置
+							ctx.translate(1248, 0); //重新定位右上角为基准
+							ctx.rotate(Math.PI / 2); //旋转90度
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							ctx.restore(); //重置画布设置
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox1)) {
+					if (this.cgs.includes(this.UIMx[page][0][1])) {
+						const img = core.material.images.images[this.UIMx[page][0][1]];
+						if (img) {
+							ctx.save(); //保存设置
+							ctx.translate(1248, 0); //重新定位右上角为基准
+							ctx.rotate(Math.PI / 2); //旋转90度
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							ctx.restore(); //重置画布设置
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox2)) {
+					if (this.cgs.includes(this.UIMx[page][0][2])) {
+						const img = core.material.images.images[this.UIMx[page][0][2]];
+						if (img) {
+							ctx.save(); //保存设置
+							ctx.translate(1248, 0); //重新定位右上角为基准
+							ctx.rotate(Math.PI / 2); //旋转90度
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							ctx.restore(); //重置画布设置
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox3)) {
+					if (this.cgs.includes(this.UIMx[page][1][0])) {
+						const img = core.material.images.images[this.UIMx[page][1][0]];
+						if (img) {
+							ctx.save(); //保存设置
+							ctx.translate(1248, 0); //重新定位右上角为基准
+							ctx.rotate(Math.PI / 2); //旋转90度
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							ctx.restore(); //重置画布设置
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox4)) {
+					if (this.cgs.includes(this.UIMx[page][1][1])) {
+						const img = core.material.images.images[this.UIMx[page][1][1]];
+						if (img) {
+							ctx.save(); //保存设置
+							ctx.translate(1248, 0); //重新定位右上角为基准
+							ctx.rotate(Math.PI / 2); //旋转90度
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							ctx.restore(); //重置画布设置
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox5)) {
+					if (this.cgs.includes(this.UIMx[page][1][2])) {
+						const img = core.material.images.images[this.UIMx[page][1][2]];
+						if (img) {
+							ctx.save(); //保存设置
+							ctx.translate(1248, 0); //重新定位右上角为基准
+							ctx.rotate(Math.PI / 2); //旋转90度
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							ctx.restore(); //重置画布设置
+							show = !show;
+						}
+					}
+				}
+			} else {
+				const pageupbox = makeBox([200, 1110], [200, 100]);
+				const pagedownbox = makeBox([1600, 1110], [200, 100]);
+				const imagebox0 = makeBox([75, 150], [600, 450]);
+				const imagebox1 = makeBox([725, 150], [600, 450]);
+				const imagebox2 = makeBox([1300, 150], [600, 450]);
+				const imagebox3 = makeBox([75, 650], [600, 450]);
+				const imagebox4 = makeBox([725, 650], [600, 450]);
+				const imagebox5 = makeBox([1375, 650], [600, 450]);
+				if (inRect(pos, pagedownbox)) {
+					if (page < this.UIMx.length - 1) {
+						page++;
+						core.clearMap(ctx);
+						this.update();
+					}
+				} else if (inRect(pos, pageupbox)) {
+					if (page > 0) {
+						page--;
+						core.clearMap(ctx);
+						this.update();
+					}
+				} else if (inRect(pos, imagebox0)) {
+					if (this.cgs.includes(this.UIMx[page][0][0])) {
+						const img = core.material.images.images[this.UIMx[page][0][0]];
+						if (img) {
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox1)) {
+					if (this.cgs.includes(this.UIMx[page][0][1])) {
+						const img = core.material.images.images[this.UIMx[page][0][1]];
+						if (img) {
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox2)) {
+					if (this.cgs.includes(this.UIMx[page][0][2])) {
+						const img = core.material.images.images[this.UIMx[page][0][2]];
+						if (img) {
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox3)) {
+					if (this.cgs.includes(this.UIMx[page][1][0])) {
+						const img = core.material.images.images[this.UIMx[page][1][0]];
+						if (img) {
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox4)) {
+					if (this.cgs.includes(this.UIMx[page][1][1])) {
+						const img = core.material.images.images[this.UIMx[page][1][1]];
+						if (img) {
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							show = !show;
+						}
+					}
+				} else if (inRect(pos, imagebox5)) {
+					if (this.cgs.includes(this.UIMx[page][1][2])) {
+						const img = core.material.images.images[this.UIMx[page][1][2]];
+						if (img) {
+							ctx.drawImage(img, 0, 0, 2028, 1248);
+							show = !show;
+						}
+					}
+				}
+			}
+		}
+		drawUI() {
+			//绘制页面
+			core.clearMap(CGUI);
+			const bgVertical = core.material.images.images["bg_2010.webp"]; //竖屏背景
+			const bg = core.material.images.images["bg_5043.webp"]; //横屏背景
 
-        if (core.domStyle.isVertical) {
-          //竖屏
+			if (core.domStyle.isVertical) {
+				//竖屏
 
-          core.fillRect(ctx, 0, 0, 1248, 2028, "#000000"); //黑色背景
-          ctx.globalAlpha = 0.5; //透明度
-          if (bgVertical)
-            ctx.drawImage(bgVertical, 0, 0, 1280, 1500, 0, 0, 1248, 2028); //绘制半透明背景图片
-          ctx.globalAlpha = 1; //恢复为不透明
+				core.fillRect(ctx, 0, 0, 1248, 2028, "#000000"); //黑色背景
+				ctx.globalAlpha = 0.5; //透明度
+				if (bgVertical)
+					ctx.drawImage(bgVertical, 0, 0, 1280, 1500, 0, 0, 1248, 2028); //绘制半透明背景图片
+				ctx.globalAlpha = 1; //恢复为不透明
 
-          core.setTextAlign(ctx, "center");
-          core.fillBoldText1(
-            ctx,
-            "◀离开",
-            100,
-            110,
-            "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
+				core.setTextAlign(ctx, "center");
+				core.fillBoldText1(
+					ctx,
+					"◀离开",
+					100,
+					110,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
 
-          core.fillBoldText1(
-            ctx,
-            "上一页",
-            300,
-            1900,
-            page === 0 ? "#444444" : "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
+				core.fillBoldText1(
+					ctx,
+					"上一页",
+					300,
+					1900,
+					page === 0 ? "#444444" : "#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
 
-          core.fillBoldText1(
-            ctx,
-            page + 1 + "/" + this.UIMx.length,
-            650,
-            1900,
-            "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
-          core.fillBoldText1(
-            ctx,
-            "下一页",
-            1000,
-            1900,
-            page === this.UIMx.length - 1 ? "#444444" : "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
-          // 添加向上翻页和向下翻页的按钮
+				core.fillBoldText1(
+					ctx,
+					page + 1 + "/" + this.UIMx.length,
+					650,
+					1900,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					"下一页",
+					1000,
+					1900,
+					page === this.UIMx.length - 1 ? "#444444" : "#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
+				// 添加向上翻页和向下翻页的按钮
 
-          // 添加3*2个4:3的画框,及图片
-          for (let i = 0; i < 3; i++) {
-            for (let j = 0; j < 2; j++) {
-              const img = core.material.images.images[this.UIMx[page][j][i]];
-              core.strokeRect(
-                ctx,
-                50 + j * 600,
-                200 + i * 550,
-                560,
-                420,
-                "#444444",
-                5
-              );
-              if (this.cgs.includes(this.UIMx[page][j][i])) {
-                if (img)
-                  ctx.drawImage(
-                    img,
-                    50 + j * 600 + 15,
-                    200 + i * 550 + 15,
-                    560 - 30,
-                    420 - 30
-                  );
-              } else {
-                ctx.fillStyle = "#000000";
-                ctx.fillRect(
-                  50 + j * 600 + 15,
-                  200 + i * 550 + 15,
-                  560 - 30,
-                  420 - 30
-                );
-                const img = core.material.images.images["LOGO.webp"];
-                if (img)
-                  ctx.drawImage(
-                    img,
-                    50 + j * 600 + 15,
-                    200 + i * 550 + 15,
-                    560 - 30,
-                    420 - 30
-                  );
-              }
-            }
-          }
-        } else {
-          //横屏
-          core.fillRect(ctx, 0, 0, 2028, 1248, "#000000"); //黑色背景
-          ctx.globalAlpha = 0.5; //透明度
-          if (bg) ctx.drawImage(bg, 0, 0, 1280, 720, 0, 0, 2028, 1248); //绘制半透明背景图片
-          ctx.globalAlpha = 1; //恢复为不透明
-          //core.drawWindowSkin('winskin1.png', ctx, 0, 0, 2028, 1248);
-          core.setTextAlign(ctx, "center");
-          core.fillBoldText1(
-            ctx,
-            "◀离开",
-            110,
-            100,
-            "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
+				// 添加3*2个4:3的画框,及图片
+				for (let i = 0; i < 3; i++) {
+					for (let j = 0; j < 2; j++) {
+						const img = core.material.images.images[this.UIMx[page][j][i]];
+						core.strokeRect(
+							ctx,
+							50 + j * 600,
+							200 + i * 550,
+							560,
+							420,
+							"#444444",
+							5
+						);
+						if (this.cgs.includes(this.UIMx[page][j][i])) {
+							if (img)
+								ctx.drawImage(
+									img,
+									50 + j * 600 + 15,
+									200 + i * 550 + 15,
+									560 - 30,
+									420 - 30
+								);
+						} else {
+							ctx.fillStyle = "#000000";
+							ctx.fillRect(
+								50 + j * 600 + 15,
+								200 + i * 550 + 15,
+								560 - 30,
+								420 - 30
+							);
+							const img = core.material.images.images["LOGO.webp"];
+							if (img)
+								ctx.drawImage(
+									img,
+									50 + j * 600 + 15,
+									200 + i * 550 + 15,
+									560 - 30,
+									420 - 30
+								);
+						}
+					}
+				}
+			} else {
+				//横屏
+				core.fillRect(ctx, 0, 0, 2028, 1248, "#000000"); //黑色背景
+				ctx.globalAlpha = 0.5; //透明度
+				if (bg) ctx.drawImage(bg, 0, 0, 1280, 720, 0, 0, 2028, 1248); //绘制半透明背景图片
+				ctx.globalAlpha = 1; //恢复为不透明
 
-          // 添加向上翻页和向下翻页的按钮
-          core.fillBoldText1(
-            ctx,
-            "上一页",
-            300,
-            1180,
-            page === 0 ? "#444444" : "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
+				core.setTextAlign(ctx, "center");
+				core.fillBoldText1(
+					ctx,
+					"◀离开",
+					110,
+					100,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
 
-          core.fillBoldText1(
-            ctx,
-            page + 1 + "/" + this.UIMx.length,
-            1000,
-            1180,
-            "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
-          core.fillBoldText1(
-            ctx,
-            "下一页",
-            1700,
-            1180,
-            page === this.UIMx.length - 1 ? "#444444" : "#FFFFFF",
-            "#000000",
-            6,
-            core.ui._buildFont(66, true)
-          );
+				// 添加向上翻页和向下翻页的按钮
+				core.fillBoldText1(
+					ctx,
+					"上一页",
+					300,
+					1180,
+					page === 0 ? "#444444" : "#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
 
-          // 添加3*2个4:3的画框
-          for (let i = 0; i < 2; i++) {
-            for (let j = 0; j < 3; j++) {
-              core.strokeRect(
-                ctx,
-                75 + j * 650,
-                150 + i * 500,
-                600,
-                450,
-                "#444444",
-                2
-              );
-              if (this.cgs.includes(this.UIMx[page][i][j])) {
-                const img = core.material.images.images[this.UIMx[page][i][j]];
-                if (img)
-                  ctx.drawImage(
-                    img,
-                    75 + j * 650 + 15,
-                    150 + i * 500 + 15,
-                    600 - 30,
-                    450 - 30
-                  );
-              } else {
-                ctx.fillStyle = "#000000";
-                ctx.fillRect(
-                  75 + j * 650 + 15,
-                  150 + i * 500 + 15,
-                  600 - 30,
-                  450 - 30
-                );
-                const img = core.material.images.images["LOGO.webp"];
-                if (img)
-                  ctx.drawImage(
-                    img,
-                    75 + j * 650 + 15,
-                    150 + i * 500 + 15,
-                    600 - 30,
-                    450 - 30
-                  );
-              }
-            }
-          }
-        }
-      }
-    }
-    this.setcgs = function (img) {
-      const a = core.getLocalStorage("cgs", []);
-      if (img) {
-        if (!a.includes(img)) a.push(img);
-        core.setLocalStorage("cgs", a);
-      } else core.setLocalStorage("cgs");
-    };
-    core.ui.CG = new CG();
-    main.dom.CGMode.onclick = function () {
-      //点击开始页面的CG MODE进入cg回廊
-      main.core.control.checkBgm();
-      page = 0;
-      main.core.ui.CG.cgs = core.getLocalStorage("cgs", []);
-      CGUI.style.display = "block";
-      main.core.ui.CG.update();
-    };
-  },
+				core.fillBoldText1(
+					ctx,
+					page + 1 + "/" + this.UIMx.length,
+					1000,
+					1180,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					"下一页",
+					1700,
+					1180,
+					page === this.UIMx.length - 1 ? "#444444" : "#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(66, true)
+				);
+
+				// 添加3*2个4:3的画框
+				for (let i = 0; i < 2; i++) {
+					for (let j = 0; j < 3; j++) {
+						core.strokeRect(
+							ctx,
+							75 + j * 650,
+							150 + i * 500,
+							600,
+							450,
+							"#444444",
+							2
+						);
+						if (this.cgs.includes(this.UIMx[page][i][j])) {
+							const img = core.material.images.images[this.UIMx[page][i][j]];
+							if (img)
+								ctx.drawImage(
+									img,
+									75 + j * 650 + 15,
+									150 + i * 500 + 15,
+									600 - 30,
+									450 - 30
+								);
+						} else {
+							ctx.fillStyle = "#000000";
+							ctx.fillRect(
+								75 + j * 650 + 15,
+								150 + i * 500 + 15,
+								600 - 30,
+								450 - 30
+							);
+							const img = core.material.images.images["LOGO.webp"];
+							if (img)
+								ctx.drawImage(
+									img,
+									75 + j * 650 + 15,
+									150 + i * 500 + 15,
+									600 - 30,
+									450 - 30
+								);
+						}
+					}
+				}
+			}
+		}
+	}
+	this.setcgs = function (img) {
+		const a = core.getLocalStorage("cgs", []);
+		if (img) {
+			if (!a.includes(img)) a.push(img);
+			core.setLocalStorage("cgs", a);
+		} else core.setLocalStorage("cgs");
+	};
+	core.ui.CG = new CG();
+	main.dom.CGMode.onclick = function () {
+		//点击开始页面的CG MODE进入cg回廊
+		main.core.control.checkBgm();
+		page = 0;
+		main.core.ui.CG.cgs = core.getLocalStorage("cgs", []);
+		CGUI.style.display = "block";
+		main.core.ui.CG.update();
+	};
+},
     "光标设置": function () {
     // 在此增加新插件
     this.changeMouse = function (
@@ -14528,1536 +15385,4107 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
     };
   },
     "信息弹出": function () {
-    // 在此增加新插件
-    /* 弹出显示某个内容
-     * 使用方法:core.addPop(px, py, value, color, boldColor, left, jump, time, show, font, speed)
-     * 参数说明:
-     * px & py: number  弹出位置
-     * value: string  显示内容
-     * color: string  填充颜色
-     * boldColor: string  描边颜色
-     */
+	// 在此增加新插件
+	/* 弹出显示某个内容
+	 * 使用方法:core.addPop(px, py, value, color, boldColor, left, jump, time, show, font, speed)
+	 * 参数说明:
+	 * px & py: number  弹出位置
+	 * value: string  显示内容
+	 * color: string  填充颜色
+	 * boldColor: string  描边颜色
+	 */
 
-    // 默认字体
-    var fontD = "16px Verdana";
-    // 默认颜色
-    var colorD = "red";
-    // 默认描边颜色
-    var boldColorD = "black";
+	// 默认字体
+	var fontD = "16px Verdana";
+	// 默认颜色
+	var colorD = "red";
+	// 默认描边颜色
+	var boldColorD = "black";
 
-    /** 血量弹出 */
-    function pop() {
-      var ctx = core.getContextByName("pop");
-      if (!ctx)
-        ctx = core.createCanvas(
-          "pop",
-          0,
-          0,
-          core.__PIXELS__,
-          core.__PIXELS__,
-          90
-        );
-      ctx.canvas.classList.add("gameCanvas", "anti-aliasing");
-      core.clearMap(ctx);
-      core.setTextAlign("pop", "left");
-      var list = core.status.pop || [];
-      var count = 0;
-      list.forEach(function (one) {
-        // 由frame计算出dy
-        var dy = 6 - one.frame * 0.2;
-        var dx = one.speed;
-        if (one.jump) {
-          one.py -= dy;
-        }
-        if (!one.left) {
-          one.px += dx;
-        } else {
-          one.px -= dx;
-        }
-        one.frame++;
-        // 绘制
-        if (one.frame >= one.time)
-          core.setAlpha(ctx, 1 - (one.frame - one.time) / one.show);
-        else core.setAlpha(ctx, 1);
-        core.fillBoldText1(
-          ctx,
-          one.value,
-          one.px,
-          one.py,
-          one.color || "red",
-          one.boldColor || "black",
-          2,
-          one.font
-        );
-        if (one.frame >= one.time + one.show) count++;
-      });
-      if (count > 0) list.splice(0, count);
-    }
-    let now = 0;
-    if (!main.replayChecking)
-      core.registerAnimationFrame("pop", true, (temptime) => {
-        if (temptime - now > 1000 / 60) {
-          now = temptime;
-          pop();
-        }
-      });
+	/** 血量弹出 */
+	function pop() {
+		var ctx = core.getContextByName("pop");
+		if (!ctx)
+			ctx = core.createCanvas(
+				"pop",
+				0,
+				0,
+				core.__PIXELS__,
+				core.__PIXELS__,
+				90
+			);
+		ctx.canvas.classList.add("gameCanvas", "anti-aliasing");
+		core.clearMap(ctx);
+		core.setTextAlign("pop", "left");
+		var list = core.status.pop || [];
+		var count = 0;
+		list.forEach(function (one) {
+			// 由frame计算出dy
+			var dy = 6 - one.frame * 0.2;
+			var dx = one.speed;
+			if (one.jump) {
+				one.py -= dy;
+			}
+			if (!one.left) {
+				one.px += dx;
+			} else {
+				one.px -= dx;
+			}
+			one.frame++;
+			// 绘制
+			if (one.frame >= one.time)
+				core.setAlpha(ctx, 1 - (one.frame - one.time) / one.show);
+			else core.setAlpha(ctx, 1);
+			core.fillBoldText1(
+				ctx,
+				one.value,
+				one.px,
+				one.py,
+				one.color || "red",
+				one.boldColor || "black",
+				2,
+				one.font
+			);
+			if (one.frame >= one.time + one.show) count++;
+		});
+		if (count > 0) list.splice(0, count);
+	}
+	if (!main.replayChecking)
+		core.registerAnimationFrame("pop", true, (temptime) => {
+			pop();
 
-    /** 添加弹出内容 */
-    this.addPop = function (
-      value,
-      px,
-      py,
-      color,
-      boldColor,
-      left,
-      jump,
-      time,
-      show,
-      font,
-      speed
-    ) {
-      var data = {
-        px: px,
-        py: py,
-        value: value,
-        color: color || colorD,
-        boldColor: boldColor || boldColorD,
-        frame: 0,
-        left: left || false,
-        jump: jump || false,
-        time: time || 60,
-        show: show || 30,
-        font: font || fontD,
-        speed: speed || 1,
-      };
-      if (!core.status.pop) core.status.pop = [data];
-      else core.status.pop.push(data);
-    };
-  },
+		});
+
+	/** 添加弹出内容 */
+	this.addPop = function (
+		value,
+		px,
+		py,
+		color,
+		boldColor,
+		left,
+		jump,
+		time,
+		show,
+		font,
+		speed
+	) {
+		var data = {
+			px: px,
+			py: py,
+			value: value,
+			color: color || colorD,
+			boldColor: boldColor || boldColorD,
+			frame: 0,
+			left: left || false,
+			jump: jump || false,
+			time: time || 60,
+			show: show || 30,
+			font: font || fontD,
+			speed: speed || 1,
+		};
+		if (!core.status.pop) core.status.pop = [data];
+		else core.status.pop.push(data);
+	};
+},
     "warning": function () {
-    // 在此增加新插件
-    // 默认音效名
-    var defaultSound = "jingbao.opus";
-    // 默认字体名
-    var defaultFont = "Verdana";
+	// 默认音效名
+	var defaultSound = "jingbao.mp3";
+	// 默认字体名
+	var defaultFont = "Verdana";
 
-    var timeout;
-    /** warning提示
-     * @param {number} x 横坐标
-     * @param {number} y 纵坐标
-     * @param {string} text 显示的文字
-     */
-    this.drawWarning = function (x, y, size, text, text2, warning) {
-      if (timeout) return;
-      x = x ?? 6;
-      y = y ?? 6;
-      text = text || "boss";
-      text += "</br>";
-      for (var i = 0; i < 10; i++) text += " ";
-      text += text2;
-      // 生成文字
-      var elements = document.querySelectorAll(".gameCanvas");
-      var t = document.createElement("p");
-      t.innerHTML = text;
-      t.style.position = "absolute";
-      t.style.fontSize = size * core.domStyle.scale + "px";
-      t.style.left = -(300 * core.domStyle.scale) + "px";
-      t.style.top = parseInt(elements[0].style.height) / 2 - 100 + "px";
-      t.style.zIndex = "300";
-      t.style.color = "#f11";
-      t.style.fontFamily = defaultFont;
-      t.style.overflow = "none";
-      t.style.width = "100%";
-      t.classList.add("warning");
-      core.dom.gameDraw.appendChild(t);
-      setTimeout(function () {
-        t.style.left = 416 * core.domStyle.scale + "px";
-      }, 50);
-      // 计算偏移量
-      var px = ((6 - x) / 12) * 50;
-      var py = ((6 - y) / 12) * 50;
-      // 修改画布的scale和transform
-      elements.forEach(function (v) {
-        if (v instanceof HTMLCanvasElement) {
-          v.style.transform = "scale(2)translate(" + px + "%, " + py + "%)";
-        }
-      });
-      if (!warning) core.playSound(defaultSound);
-      // 拉回镜头
-      timeout = setTimeout(function () {
-        timeout = setTimeout(function () {
-          timeout = void 0;
-          core.dom.gameDraw.removeChild(t);
-        }, 1500);
-        elements.forEach(function (v) {
-          if (v instanceof HTMLCanvasElement) {
-            v.style.transform = "none";
-          }
-        });
-      }, 1600);
-    };
-  },
-    "立体声音效": function () {
-    // 在此增加新插件
-    // 音效双声道播放
-    var can = true;
-    if (!AudioContext) {
-      console.warn("该浏览器不支持AudioContext,无法播放立体声");
-      can = false;
-    }
-    if (can) var ac = new AudioContext();
-    var datas = {};
-
-    /** 播放立体声 参考:https://developer.mozilla.org/zh-CN/docs/Web/API/BaseAudioContext/createChannelSplitter
-     * @param {number} name 音效名
-     * @param {number} left 左声道音量,默认为1
-     * @param {number} right 右声道音量,默认为1
-     * @param {boolean} split 音效为双声道,请填true,为单声道,请填false或不填
-     * @returns 该音效的唯一id
-     */
-    this.playStereo = function (name, left, right, split) {
-      if (!can) return core.playSound(name);
-      var sound = core.getMappedName(name);
-      if (
-        main.mode != "play" ||
-        !core.musicStatus.soundStatus ||
-        !core.material.sounds[sound]
-      )
-        return;
-      if (!core.status.stereo) core.status.stereo = {};
-      var buffer = core.material.sounds[sound];
-      var source = ac.createBufferSource();
-      source.buffer = buffer;
-      var splitter = ac.createChannelSplitter(2);
-      source.connect(splitter);
-      var merger = ac.createChannelMerger(2);
-
-      /* 大致流程
-		                            gain(L)
-		                        /			  \
-		source  ----  splitter  				 merger  ----  destination
-		                        \			  /
-		                            gain(R) 
-		*/
-
-      var L = ac.createGain();
-      var R = ac.createGain();
-      L.gain.value = left * core.musicStatus.userVolume;
-      R.gain.value = right * core.musicStatus.userVolume;
-      splitter.connect(L, 0, 0);
-      if (!split) splitter.connect(R, 0, 0);
-      else splitter.connect(R, 1, 0);
-
-      var id = setTimeout(null);
-      core.status.stereo[id] = { source: source, L: L, R: R };
-
-      L.connect(merger, 0, 0);
-      R.connect(merger, 0, 1);
-
-      var dest = ac.destination;
-
-      source.onended = function () {
-        delete datas[id];
-        source = void 0;
-      };
-
-      merger.connect(dest);
-      source.start(0);
-      return id;
-    };
-
-    /** 移动声源(渐变调整左右声道音量)
-     * @param {number} id 为playStereo返回的id
-     * @param {number} left 要渐变到的左声道音量
-     * @param {number} right 要渐变到的右声道音量
-     * @param {number} time 渐变时间
-     */
-    this.moveStereo = function (id, left, right, time) {
-      if (!can) return;
-      if (main.mode != "play" || !core.musicStatus.soundStatus) return;
-      var stereo = core.status.stereo[id];
-      datas[id] = {
-        time: time,
-        curr: 0,
-        dL:
-          ((left - stereo.L.gain.value) / time) *
-          10 *
-          core.musicStatus.userVolume,
-        dR:
-          ((right - stereo.R.gain.value) / time) *
-          10 *
-          core.musicStatus.userVolume,
-      };
-      var interval = setInterval(function () {
-        var data = datas[id];
-        if (!data) return clearInterval(interval);
-        data.curr += 10;
-        if (data.curr >= data.time) {
-          clearInterval(interval);
-          datas[id] = void 0;
-        }
-        stereo.L.gain.value += data.dL;
-        stereo.R.gain.value += data.dR;
-      }, 10);
-    };
-  },
+	var timeout;
+	/** warning提示
+	 * @param {number} x 横坐标
+	 * @param {number} y 纵坐标
+	 * @param {string} text 显示的文字
+	 */
+	this.drawWarning = function (
+		x,
+		y,
+		text,
+		text2,
+		warning,
+		large = 2,
+		size = 36
+	) {
+		if (timeout) return;
+		x = x ?? 6;
+		y = y ?? 6;
+		text = text || "boss";
+		text += "</br>";
+		for (var i = 0; i < 10; i++) text += " ";
+		text += text2;
+		// 生成文字
+		var elements = document.querySelectorAll(".gameCanvas");
+		var t = document.createElement("p");
+		t.innerHTML = text;
+		t.style.position = "absolute";
+		t.style.fontSize = size * core.domStyle.scale + "px";
+		t.style.left = -(300 * core.domStyle.scale) + "px";
+		t.style.top = parseInt(elements[0].style.height) / 2 - 100 + "px";
+		t.style.zIndex = "300";
+		t.style.color = "#f11";
+		t.style.fontFamily = defaultFont;
+		t.style.overflow = "none";
+		t.style.width = "100%";
+		t.classList.add("warning");
+		core.dom.gameDraw.appendChild(t);
+		setTimeout(function () {
+			t.style.left = 416 * core.domStyle.scale + "px";
+		}, 50);
+		// 计算偏移量
+		var px = ((6 - x) / 12) * (100 - 100 / large + 2);
+		var py = ((6 - y) / 12) * (100 - 100 / large + 2);
+		// 修改画布的scale和transform
+		let time = 0;
+		let s = 1;
+		let sx = 0;
+		let sy = 0;
+		let cishu = 1;
+		core.registerAnimationFrame("big", true, function (temptime) {
+			if (temptime - time > 10) {
+				time = temptime;
+				s += (large - 1) / 30;
+				sx += px / 30;
+				sy += py / 30;
+				elements.forEach(function (v) {
+					if (v instanceof HTMLCanvasElement) {
+						v.style.transform =
+							"scale(" + s + ")translate(" + sx + "%, " + sy + "%)";
+					}
+				});
+				cishu++;
+				if (cishu == 30) {
+					core.unregisterAnimationFrame("big");
+				}
+			}
+		});
+		if (!warning) core.playSound(defaultSound);
+		// 拉回镜头
+		timeout = setTimeout(function () {
+			// 			timeout = setTimeout(function () {
+			// 				timeout = void 0;
+			// 				core.dom.gameDraw.removeChild(t);
+			// 			}, 1500);
+			let time2 = 0;
+			let s2 = large - (large - 1) / 30;
+			let sx2 = px;
+			let sy2 = py;
+			let cishu2 = 1;
+			core.registerAnimationFrame("small", true, function (temptime) {
+				if (temptime - time2 > 10) {
+					time2 = temptime;
+					s2 -= (large - 1) / 30;
+					sx2 -= px / 30;
+					sy2 -= py / 30;
+					elements.forEach(function (v) {
+						if (v instanceof HTMLCanvasElement) {
+							v.style.transform =
+								"scale(" + s2 + ")translate(" + sx2 + "%, " + sy2 + "%)";
+						}
+					});
+					cishu2++;
+					if (cishu2 == 30) {
+						core.unregisterAnimationFrame("small");
+						elements.forEach(function (v) {
+							if (v instanceof HTMLCanvasElement) {
+								v.style.transform = "none";
+							}
+						});
+					}
+				}
+			});
+			// 			elements.forEach(function (v) {
+			// 				if (v instanceof HTMLCanvasElement) {
+			// 					v.style.transform = "none";
+			// 				}
+			// 			});
+		}, 1600);
+	};
+},
     "滑动转场": function () {
-    // 在此增加新插件
-    const defaultChange = {
-      left: "leftPortal", // 左箭头
-      up: "upPortal", // 上箭头
-      right: "rightPortal", // 右箭头
-      down: "downPortal", // 下箭头
-      upFloor: "upFloor", // 上楼
-      downFloor: "downFloor", // 下楼
-    };
-    const dirData = {
-      //方向坐标
-      up: [-1, 0],
-      down: [1, 0],
-      left: [0, -1],
-      right: [0, 1],
-      upFloor: [0, 0],
-      downFloor: [0, 0],
-    };
-    let allChangeEntries = Object.entries(defaultChange);
-    const move = document.createElement("canvas");
-    const speed = 12;
-    let modedata = 0;
+	// 在此增加新插件
+	const defaultChange = {
+		left: "leftPortal", // 左箭头
+		up: "upPortal", // 上箭头
+		right: "rightPortal", // 右箭头
+		down: "downPortal", // 下箭头
+		upFloor: "upFloor", // 上楼
+		downFloor: "downFloor", // 下楼
+	};
+	const dirData = {
+		//方向坐标
+		up: [-1, 0],
+		down: [1, 0],
+		left: [0, -1],
+		right: [0, 1],
+		upFloor: [0, 0],
+		downFloor: [0, 0],
+	};
+	let allChangeEntries = Object.entries(defaultChange);
+	const move = document.createElement("canvas");
+	const speed = 12;
+	let modedata = 0;
 
-    move.width = 1248;
-    move.height = 1248;
+	move.width = 1248;
+	move.height = 1248;
 
-    const ctx = move.getContext("2d");
+	const ctx = move.getContext("2d");
 
-    events.prototype.changeFloor = function (
-      floorId,
-      stair,
-      heroLoc,
-      time,
-      callback
-    ) {
-      let block = core.getBlock(hero.loc.x, hero.loc.y);
-      var info = this._changeFloor_getInfo(floorId, stair, heroLoc, time);
+	events.prototype.changeFloor = function (
+		floorId,
+		stair,
+		heroLoc,
+		time,
+		callback
+	) {
+		let block = core.getBlock(hero.loc.x, hero.loc.y);
+		var info = this._changeFloor_getInfo(floorId, stair, heroLoc, time);
 
-      if (info == null) {
-        if (callback) callback();
-        return;
-      }
-      floorId = info.floorId;
-      info.locked = core.status.lockControl;
+		if (info == null) {
+			if (callback) callback();
+			return;
+		}
+		floorId = info.floorId;
+		info.locked = core.status.lockControl;
 
-      core.dom.floorNameLabel.innerText = core.status.maps[floorId].title;
-      core.lockControl();
-      core.stopAutomaticRoute();
-      core.clearContinueAutomaticRoute();
-      core.status.replay.animate = true;
-      clearInterval(core.interval.onDownInterval);
-      core.interval.onDownInterval = "tmp";
+		core.dom.floorNameLabel.innerText = core.status.maps[floorId].title;
+		core.lockControl();
+		core.stopAutomaticRoute();
+		core.clearContinueAutomaticRoute();
+		core.status.replay.animate = true;
+		clearInterval(core.interval.onDownInterval);
+		delete core.animateFrame.tip;
+		core.interval.onDownInterval = "tmp";
 
-      this._changeFloor_beforeChange(info, block, callback);
-    };
-    events.prototype._changeFloor_beforeChange = function (
-      info,
-      block,
-      callback
-    ) {
-      this._changeFloor_playSound();
-      if (
-        block &&
-        block?.event &&
-        !main.replayChecking &&
-        !core.isReplaying()
-      ) {
-        const dirEntries = allChangeEntries.find(
-          (v) => v[1] === block.event.id
-        );
-        if (block?.event?.trigger === "changeFloor" && dirEntries) {
-          const toFloorId = block.event.data.floorId;
-          const dir = dirEntries[0];
+		this._changeFloor_beforeChange(info, block, callback);
+	};
+	events.prototype._changeFloor_beforeChange = function (
+		info,
+		block,
+		callback
+	) {
+		this._changeFloor_playSound();
+		if (
+			block &&
+			block?.event &&
+			!main.replayChecking &&
+			!core.isReplaying() &&
+			!core.hasFlag("__isFlying__") && !core.hasFlag("__fromLoad__")
+		) {
+			const dirEntries = allChangeEntries.find(
+				(v) => v[1] === block.event.id
+			);
+			if (block?.event?.trigger === "changeFloor" && dirEntries) {
+				const toFloorId = block.event.data.floorId;
+				const dir = dirEntries[0];
 
-          const data = core.ui._drawViewMaps_buildData(
-            core.floorIds.indexOf(core.status.floorId)
-          );
-          const dataTo = core.ui._drawViewMaps_buildData(
-            core.floorIds.indexOf(toFloorId)
-          );
-          const v = dirData[dir][1], // 水平数值
-            h = dirData[dir][0]; //竖直数值
-          ctx.clearRect(0, 0, 1248, 1248);
-          core.drawThumbnail(core.status.floorId, null, {
-            damage: data.damage,
-            ctx: ctx,
-            x: 416,
-            y: 416,
-            size: 1,
-            all: data.all,
-          });
-          if (dir !== "upFloor" && dir !== "downFloor") {
-            core.drawThumbnail(toFloorId, null, {
-              damage: dataTo.damage,
-              ctx: ctx,
-              x: 416 + 416 * v,
-              y: 416 + 416 * h,
-              size: 1,
-              all: dataTo.all,
-            });
+				const data = core.ui._drawViewMaps_buildData(
+					core.floorIds.indexOf(core.status.floorId)
+				);
+				const dataTo = core.ui._drawViewMaps_buildData(
+					core.floorIds.indexOf(toFloorId)
+				);
+				const v = dirData[dir][1], // 水平数值
+					h = dirData[dir][0]; //竖直数值
+				ctx.clearRect(0, 0, 1248, 1248);
+				core.drawThumbnail(core.status.floorId, null, {
+					damage: data.damage,
+					ctx: ctx,
+					x: 416,
+					y: 416,
+					size: 1,
+					all: data.all,
+				});
+				if (dir !== "upFloor" && dir !== "downFloor") {
+					core.drawThumbnail(toFloorId, null, {
+						damage: dataTo.damage,
+						ctx: ctx,
+						x: 416 + 416 * v,
+						y: 416 + 416 * h,
+						size: 1,
+						all: dataTo.all,
+					});
 
-            var _run = function () {
-              var cb = function () {
-                modedata = 0;
-                core.clearUI();
-                core.clearMap("data");
-                core.events._changeFloor_changing(info, callback);
-              };
+					var _run = function () {
+						var cb = function () {
+							modedata = 0;
+							core.clearUI();
+							core.clearMap("data");
+							core.events._changeFloor_changing(info, callback);
+						};
 
-              var animate = window.setInterval(
-                function () {
-                  if (modedata >= 416) {
-                    delete core.animateFrame.asyncId[animate];
-                    clearInterval(animate);
-                    cb();
-                  } else {
-                    core.clearUI();
-                    core.clearMap("data");
+						var animate = window.setInterval(
+							function () {
+								if (modedata >= 416) {
+									delete core.animateFrame.asyncId[animate];
+									clearInterval(animate);
+									cb();
+								} else {
+									core.clearUI();
+									core.clearMap("data");
 
-                    core.canvas.data.drawImage(
-                      move,
-                      416 + modedata * v,
-                      416 + modedata * h,
-                      416,
-                      416,
-                      0,
-                      0,
-                      416,
-                      416
-                    );
-                    let status = "leftFoot";
+									core.canvas.data.drawImage(
+										move,
+										416 + modedata * v,
+										416 + modedata * h,
+										416,
+										416,
+										0,
+										0,
+										416,
+										416
+									);
+									let status = "leftFoot";
 
-                    if (modedata > 208) {
-                      status = "rightFoot";
-                    }
-                    const img = core.material.images.hero;
-                    const heroIconArr = core.material.icons.hero;
-                    const width = core.material.icons.hero.width || 32;
-                    const height = core.material.icons.hero.height;
-                    const heroIcon = heroIconArr[dir];
+									if (modedata > 208) {
+										status = "rightFoot";
+									}
+									const img = core.material.images.hero;
+									const heroIconArr = core.material.icons.hero;
+									const width = core.material.icons.hero.width || 32;
+									const height = core.material.icons.hero.height;
+									const heroIcon = heroIconArr[dir];
 
-                    core.canvas.data.drawImage(
-                      img,
-                      (heroIcon[status] % 4) * width,
-                      heroIcon.loc * height,
-                      width,
-                      height,
-                      core.status.hero.loc.x * 32 -
-                        core.bigmap.offsetX -
-                        (modedata - (modedata * 32) / 416) * v,
-                      core.status.hero.loc.y * 32 -
-                        16 -
-                        (modedata - (modedata * 32) / 416) * h,
-                      width,
-                      height
-                    );
-                    modedata += speed;
-                    clearInterval(animate);
-                    delete core.animateFrame.asyncId[animate];
-                    _run();
-                  }
-                },
-                core.status.replay.speed == 24
-                  ? 1
-                  : 10 / core.status.replay.speed
-              );
+									core.canvas.data.drawImage(
+										img,
+										(heroIcon[status] % 4) * width,
+										heroIcon.loc * height,
+										width,
+										height,
+										core.status.hero.loc.x * 32 -
+										core.bigmap.offsetX -
+										(modedata - (modedata * 32) / 416) * v,
+										core.status.hero.loc.y * 32 -
+										16 -
+										(modedata - (modedata * 32) / 416) * h,
+										width,
+										height
+									);
+									modedata += speed;
+									clearInterval(animate);
+									delete core.animateFrame.asyncId[animate];
+									_run();
+								}
+							},
+							core.status.replay.speed == 24 ?
+							1 :
+							10 / core.status.replay.speed
+						);
 
-              core.animateFrame.lastAsyncId = animate;
-              core.animateFrame.asyncId[animate] = cb;
-            };
-            _run();
-            return;
-          }
-        }
-      }
-      // 需要 setTimeout 执行,不然会出错
-      window.setTimeout(function () {
-        if (info.time == 0) core.events._changeFloor_changing(info, callback);
-        else
-          core.showWithAnimate(
-            core.dom.floorMsgGroup,
-            info.time / 2,
-            function () {
-              core.events._changeFloor_changing(info, callback);
-            }
-          );
-      }, 25);
-    };
-  },
+						core.animateFrame.lastAsyncId = animate;
+						core.animateFrame.asyncId[animate] = cb;
+					};
+					_run();
+					return;
+				}
+			}
+		}
+		// 需要 setTimeout 执行,不然会出错
+		window.setTimeout(function () {
+			if (info.time == 0) core.events._changeFloor_changing(info, callback);
+			else
+				core.showWithAnimate(
+					core.dom.floorMsgGroup,
+					info.time / 2,
+					function () {
+						core.events._changeFloor_changing(info, callback);
+					}
+				);
+		}, 25);
+	};
+},
     "剧情cg": function () {
-    // 在此增加新插件
-    const cg = document.createElement("canvas"); //cg画布设置
-    cg.style.position = "absolute";
-    cg.style.zIndex = 300;
-    cg.style.display = "none";
-    cg.id = "cgText";
-    main.dom.gameGroup.insertAdjacentElement("afterend", cg);
-    cg.style.top = "50%";
-    cg.style.left = "50%";
-    cg.style.transform = "translate(-50%,-50%)";
-    const ctx = cg.getContext("2d");
-    main.dom.cgText = cg;
+	// 在此增加新插件
+	// 在此增加新插件
+	const cg = document.createElement("canvas"); //cg画布设置
+	cg.style.position = "absolute";
+	cg.style.zIndex = 320;
+	cg.style.display = "none";
+	cg.id = "cgText";
+	main.dom.gameGroup.insertAdjacentElement("afterend", cg);
+	cg.style.top = "50%";
+	cg.style.left = "50%";
+	cg.style.transform = "translate(-50%,-50%)";
+	const ctx = cg.getContext("2d");
+	main.dom.cgText = cg;
+	const logcanvas = document.createElement("canvas"); //cg画布设置
+	logcanvas.style.position = "absolute";
+	logcanvas.style.zIndex = 321;
+	logcanvas.style.display = "none";
+	logcanvas.id = "cgText";
+	main.dom.gameGroup.insertAdjacentElement("afterend", logcanvas);
+	logcanvas.style.top = "50%";
+	logcanvas.style.left = "50%";
+	logcanvas.style.transform = "translate(-50%,-50%)";
+	const logctx = logcanvas.getContext("2d");
+	main.dom.logcanvas = logcanvas;
+	logcanvas.onmouseup = function (e) {
+		//鼠标抬起
 
-    cg.onmouseup = function (e) {
-      //鼠标抬起
+		try {
+			if (!core.isPlaying()) return false;
+			core.unregisterAnimationFrame("skip");
+			let a = core.getFlag("skip", false);
+			core.setFlag("skip", false);
+			if (a) {
+				const data = core.clone(core.status.event.data.current);
 
-      try {
-        e.stopPropagation();
-        if (!core.isPlaying()) return false;
-        core.unregisterAnimationFrame("skip");
-        let a = core.getFlag("skip", false);
-        core.setFlag("skip", false);
-        if (a) {
-          const data = core.clone(core.status.event.data.current);
+				core.insertAction(data);
+				core.doAction();
+			}
+		} catch (ee) {
+			console.error(ee);
+		}
+	};
+	logcanvas.onmousedown = function (e) {
+		//鼠标按下
+		try {
+			if (!core.isPlaying()) return false;
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+				py = Math.floor((e.clientY - top) / core.domStyle.scale);
+			core.ui.cgText.click(px * 3, py * 3);
+		} catch (ee) {
+			main.log(ee);
+		}
+	};
+	logcanvas.ontouchend = function (e) {
+		//触摸抬起
 
-          core.insertAction(data);
-          core.doAction();
-        }
-      } catch (ee) {
-        console.error(ee);
-      }
-    };
-    cg.onmousedown = function (e) {
-      //鼠标按下
-      try {
-        e.stopPropagation();
-        if (!core.isPlaying()) return false;
-        const left = core.dom.gameGroup.offsetLeft;
-        const top = core.dom.gameGroup.offsetTop;
-        const px = Math.floor((e.clientX - left) / core.domStyle.scale),
-          py = Math.floor((e.clientY - top) / core.domStyle.scale);
-        core.ui.cgText.click(px * 3, py * 3);
-      } catch (ee) {
-        main.log(ee);
-      }
-    };
-    cg.ontouchend = function (e) {
-      //触摸抬起
+		try {
+			if (!core.isPlaying()) return false;
+			core.unregisterAnimationFrame("skip");
+			core.setFlag("skip", false);
+			let a = core.getFlag("skip", false);
+			core.setFlag("skip", false);
+			if (a) {
+				const data = core.clone(core.status.event.data.current);
 
-      try {
-        e.preventDefault();
-        if (!core.isPlaying()) return false;
-        core.unregisterAnimationFrame("skip");
-        core.setFlag("skip", false);
-        let a = core.getFlag("skip", false);
-        core.setFlag("skip", false);
-        if (a) {
-          const data = core.clone(core.status.event.data.current);
+				core.insertAction(data);
+				core.doAction();
+			}
+		} catch (ee) {}
+	};
+	logcanvas.ontouchstart = function (e) {
+		//触摸按下
+		try {
+			if (!core.isPlaying()) return false;
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor(
+					(e.targetTouches[0].clientX - left) / core.domStyle.scale
+				),
+				py = Math.floor(
+					(e.targetTouches[0].clientY - top) / core.domStyle.scale
+				);
+			core.ui.cgText.click(px * 3, py * 3);
+		} catch (ee) {
+			main.log(ee);
+		}
+	};
+	cg.onmouseup = function (e) {
+		//鼠标抬起
 
-          core.insertAction(data);
-          core.doAction();
-        }
-      } catch (ee) {
-        console.error(ee);
-      }
-    };
-    cg.ontouchstart = function (e) {
-      //触摸按下
-      try {
-        e.preventDefault();
-        if (!core.isPlaying()) return false;
-        const left = core.dom.gameGroup.offsetLeft;
-        const top = core.dom.gameGroup.offsetTop;
-        const px = Math.floor(
-            (e.targetTouches[0].clientX - left) / core.domStyle.scale
-          ),
-          py = Math.floor(
-            (e.targetTouches[0].clientY - top) / core.domStyle.scale
-          );
-        core.ui.cgText.click(px * 3, py * 3);
-      } catch (ee) {
-        main.log(ee);
-      }
-    };
-    let auto = false;
-    class cgText {
-      constructor() {
-        //绘制需要的变量
-        this.image = "";
-        this.head = { name: "face_050445.webp", px: -300 };
-        this.bodyList = [
-          { name: "tati_050145a.webp", px: 100, filter: false },
-          { name: "tati_120124.webp", px: 1100, filter: true },
-        ];
-        this.name = "";
-        this.text = "";
-        this.time = 0;
-        this.WindowSkin = false;
-        this.sound = "";
-        this.beforeSound = 0;
-        this.wait = 200;
-        this.memory = false;
-      }
-      click(px, py) {
-        //点击效果
+		try {
+			if (!core.isPlaying()) return false;
+			core.unregisterAnimationFrame("skip");
+			let a = core.getFlag("skip", false);
+			core.setFlag("skip", false);
+			if (a) {
+				const data = core.clone(core.status.event.data.current);
 
-        const makeBox = ([x, y], [w, h]) => {
-          return [
-            [x, y],
-            [x + w, y + h],
-          ];
-        };
-        const inRect = ([x, y], [[sx, sy], [dx, dy]]) => {
-          return sx <= x && x <= dx && sy <= y && y <= dy;
-        };
-        const pos = [px, py];
-        const savebox = makeBox([1700, 1100], [192, 96]);
-        const saveboxVertical = makeBox([52, 1700], [96, 192]);
-        const skipbox = makeBox([1400, 1100], [192, 96]);
-        const skipboxVertical = makeBox([52, 1400], [96, 192]);
-        const autobox = makeBox([1700, 900], [192, 96]);
-        const autoboxVertical = makeBox([256, 1700], [96, 192]);
-        if (
-          (core.domStyle.isVertical &&
-            inRect(pos, skipboxVertical) &&
-            !this.WindowSkin) ||
-          (!core.domStyle.isVertical &&
-            !this.WindowSkin &&
-            inRect(pos, skipbox))
-        ) {
-          auto = false;
-          let time = 0;
-          core.stopSound(this.beforeSound);
-          core.registerAnimationFrame("skip", true, (timestamp) => {
-            if (timestamp > time + 50) {
-              time = timestamp;
-              if (
-                core.status.event.id == "action" &&
-                core.status.event.data.type == "cgtext"
-              ) {
-                core.setFlag("skip", true);
-                main.dom.cgText.style.display = "none";
-                core.doAction();
-              }
-            }
-          });
-        } else if (
-          (core.domStyle.isVertical &&
-            inRect(pos, autoboxVertical) &&
-            !this.WindowSkin) ||
-          (!core.domStyle.isVertical &&
-            !this.WindowSkin &&
-            inRect(pos, autobox))
-        ) {
-          auto = !auto;
+				core.insertAction(data);
+				core.doAction();
+			}
+		} catch (ee) {
+			console.error(ee);
+		}
+	};
+	cg.onmousedown = function (e) {
+		//鼠标按下
+		try {
+			if (!core.isPlaying()) return false;
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+				py = Math.floor((e.clientY - top) / core.domStyle.scale);
+			core.ui.cgText.click(px * 3, py * 3);
+		} catch (ee) {
+			main.log(ee);
+		}
+	};
+	cg.ontouchend = function (e) {
+		//触摸抬起
 
-          const data = core.clone(core.status.event.data.current);
-          data.showAll = true;
-          data.time = 0;
-          data.text = data.text.replace(/(\\(z))(\[.*?])?/g, ""); //去除打字机暂停效果
-          data.sound = "";
-          core.insertAction(data);
-          core.doAction();
-        } else if (
-          (core.domStyle.isVertical &&
-            inRect(pos, saveboxVertical) &&
-            !this.WindowSkin) ||
-          (!core.domStyle.isVertical &&
-            !this.WindowSkin &&
-            inRect(pos, savebox))
-        ) {
-          //存档
-          auto = false;
-          if (core.status.event.animateUI) return;
-          if (core.status.event.interval != null) return;
-          const current = core.clone(core.status.event.data.current);
-          current.showAll = true;
-          current.time = 0;
-          current.sound = "";
-          current.text = current.text.replace(/(\\(z))(\[.*?])?/g, ""); //去除当前事件所有打字机效果
-          cg.style.display = "none";
-          const data = [{ type: "callSave" }, current]; //插入存档事件
-          core.insertAction(data);
+		try {
+			if (!core.isPlaying()) return false;
+			core.unregisterAnimationFrame("skip");
+			core.setFlag("skip", false);
+			let a = core.getFlag("skip", false);
+			core.setFlag("skip", false);
+			if (a) {
+				const data = core.clone(core.status.event.data.current);
 
-          core.doAction();
-        } else if (!core.status.event.data) {
-          cg.style.display = "none";
-          core.ui._animateUI("hide", null, () => {
-            core.doAction();
-          });
-        } else {
-          // 正在淡入淡出的话不执行
-          if (core.status.event.animateUI) return;
-          auto = false;
+				core.insertAction(data);
+				core.doAction();
+			}
+		} catch (ee) {
+			console.error(ee);
+		}
+	};
+	cg.ontouchstart = function (e) {
+		//触摸按下
+		try {
+			if (!core.isPlaying()) return false;
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor(
+					(e.targetTouches[0].clientX - left) / core.domStyle.scale
+				),
+				py = Math.floor(
+					(e.targetTouches[0].clientY - top) / core.domStyle.scale
+				);
+			core.ui.cgText.click(px * 3, py * 3);
+		} catch (ee) {
+			main.log(ee);
+		}
+	};
+	let auto = false;
 
-          // 打字机效果显示全部文字
-          if (core.status.event.interval != null) {
-            const data = core.clone(core.status.event.data?.current);
-            data.showAll = true;
-            data.time = 0;
-            data.text = data.text.replace(/(\\(z))(\[.*?])?/g, ""); //去除打字机暂停效果
-            data.sound = "";
-            core.insertAction(data);
-            core.doAction();
-            return;
-          } else {
-            core.stopSound(this.beforeSound);
-          }
+	class cgText {
+		constructor() {
+			this.nobg = false;
+			//绘制需要的变量
+			this.image = "";
+			this.head = { name: "face_050445.webp", px: -300 };
+			this.bodyList = [
+				{ name: "tati_050145a.webp", px: 100, filter: false },
+				{ name: "tati_120124.webp", px: 1100, filter: true },
+			];
+			this.name = "";
+			this.text = "";
+			this.time = 0;
+			this.WindowSkin = false;
+			this.sound = "";
+			this.beforeSound = 0;
+			this.wait = 1000;
+			this.memory = false;
+			this.textList = "";
+			this.page = 1;
+			this.overpage = 1;
+			this.log = false;
+			this.index = 0;
+		}
+		click(px, py) {
+			//点击效果
 
-          cg.style.display = "none";
-          core.ui._animateUI("hide", null, () => {
-            core.doAction();
-          });
-        }
-      }
+			const makeBox = ([x, y], [w, h]) => {
+				return [
+					[x, y],
+					[x + w, y + h],
+				];
+			};
+			const inRect = ([x, y], [
+				[sx, sy],
+				[dx, dy]
+			]) => {
+				return sx <= x && x <= dx && sy <= y && y <= dy;
+			};
+			const pos = [px, py];
+			const savebox = makeBox([1700, 1100], [192, 96]);
+			const saveboxVertical = makeBox([52, 1700], [96, 192]);
+			const skipbox = makeBox([1700, 1000], [192, 96]);
+			const skipboxVertical = makeBox([152, 1700], [96, 192]);
+			const autobox = makeBox([1700, 900], [192, 96]);
+			const autoboxVertical = makeBox([252, 1700], [96, 192]);
+			const textbox = makeBox([1700, 800], [192, 96]);
+			const textboxVertical = makeBox([352, 1700], [96, 192]);
+			const backbox = makeBox([15, 35], [210, 90]);
+			const backboxVertical = makeBox([1123, 15], [90, 210]);
+			const pageupbox = makeBox([300, 1130], [200, 100]);
+			const pageupboxVertical = makeBox([18, 300], [100, 200]);
+			const pagedownbox = makeBox([1500, 1130], [200, 100]);
+			const pagedownboxVertical = makeBox([18, 1500], [100, 200]);
+			const soundbox = makeBox([550, 150], [100, 900]);
+			const soundboxVertical = makeBox([198, 550], [900, 100]);
+			if (this.log) {
+				if (
+					(core.domStyle.isVertical && inRect(pos, backboxVertical)) ||
+					(!core.domStyle.isVertical && inRect(pos, backbox))
+				) {
+					core.clearMap(logctx);
+					core.stopSound();
+					main.dom.logcanvas.style.display = "none";
+					this.log = false;
+				} else if (
+					(core.domStyle.isVertical && inRect(pos, pageupboxVertical)) ||
+					(!core.domStyle.isVertical && inRect(pos, pageupbox))
+				) {
+					core.clearMap(logctx);
+					if (this.page > 1) this.page--;
+					this.logdraw(this.page);
+				} else if (
+					(core.domStyle.isVertical && inRect(pos, pagedownboxVertical)) ||
+					(!core.domStyle.isVertical && inRect(pos, pagedownbox))
+				) {
+					core.clearMap(logctx);
+					if (this.page < this.overpage) this.page++;
+					this.logdraw(this.page);
+				} else if (
+					(core.domStyle.isVertical && inRect(pos, soundboxVertical)) ||
+					(!core.domStyle.isVertical && inRect(pos, soundbox))
+				) {
+					if (core.domStyle.isVertical) {
+						const sound =
+							core.plugin[this.textList][
+								(this.page - 1) * 6 +
+								Math.min(Math.floor((px - 198) / 150), 5)
+							][2];
+						core.stopSound();
+						core.playSound(sound);
+					} else {
+						const sound =
+							core.plugin[this.textList][
+								(this.page - 1) * 6 +
+								Math.min(Math.floor((py - 150) / 150), 5)
+							][2];
+						core.stopSound();
+						core.playSound(sound);
+					}
+				}
+			} else {
+				if (
+					(core.domStyle.isVertical &&
+						inRect(pos, skipboxVertical) &&
+						!this.WindowSkin) ||
+					(!core.domStyle.isVertical &&
+						!this.WindowSkin &&
+						inRect(pos, skipbox))
+				) {
+					auto = false;
+					let time = 0;
+					core.stopSound(this.beforeSound);
+					core.registerAnimationFrame("skip", true, (timestamp) => {
+						if (timestamp > time + 50) {
+							time = timestamp;
+							if (
+								core.status.event.id == "action" &&
+								core.status.event.data.type == "cgtext"
+							) {
+								core.setFlag("skip", true);
+								main.dom.cgText.style.display = "none";
+								core.doAction();
+							}
+						}
+					});
+				} else if (
+					(core.domStyle.isVertical &&
+						inRect(pos, textboxVertical) &&
+						!this.WindowSkin) ||
+					(!core.domStyle.isVertical &&
+						!this.WindowSkin &&
+						inRect(pos, textbox))
+				) {
+					auto = false;
+					this.log = true;
+					this.overpage = Math.floor(this.index / 6) + 1;
+					this.page = this.overpage;
+					logcanvas.style.display = "block";
+					const data = core.clone(core.status.event.data.current);
+					data.showAll = true;
+					data.time = 0;
+					data.text = this.text.replaceAll(/(\\(z))(\[.*?\])?/g, ""); //去除打字机暂停效果
+					data.sound = "";
+					core.insertAction(data);
+					core.doAction();
+					this.logdraw(this.page);
+				} else if (
+					(core.domStyle.isVertical &&
+						inRect(pos, autoboxVertical) &&
+						!this.WindowSkin) ||
+					(!core.domStyle.isVertical &&
+						!this.WindowSkin &&
+						inRect(pos, autobox))
+				) {
+					auto = !auto;
 
-      drawTextContent(ctx, content, config) {
-        //绘制多行文字并执行打字机效果
+					const data = core.clone(core.status.event.data.current);
+					data.showAll = true;
+					data.time = 0;
+					data.text = this.text.replaceAll(/(\\(z))(\[.*?\])?/g, ""); //去除打字机暂停效果
+					data.sound = "";
+					core.insertAction(data);
+					core.doAction();
+				} else if (
+					(core.domStyle.isVertical &&
+						inRect(pos, saveboxVertical) &&
+						!this.WindowSkin) ||
+					(!core.domStyle.isVertical &&
+						!this.WindowSkin &&
+						inRect(pos, savebox))
+				) {
+					//存档
+					auto = false;
+					if (core.status.event.animateUI) return;
+					if (core.status.event.interval != null) return;
+					const current = core.clone(core.status.event.data.current);
+					current.showAll = true;
+					current.time = 0;
+					current.sound = "";
+					current.text = this.text.replaceAll(/(\\(z))(\[.*?\])?/g, ""); //去除当前事件所有打字机效果
+					cg.style.display = "none";
+					core.setFlag("_cgText", true)
+					const data = [{ type: "callSave" }, { "type": "setValue", "name": "flag:_cgText", "value": "false" }, current]; //插入存档事件
+					core.insertAction(data);
 
-        ctx = core.getContextByName(ctx);
-        // 设置默认配置项
-        var textAttribute =
-          core.status.textAttribute || core.initStatus.textAttribute;
-        var globalAttribute =
-          core.status.globalAttribute || core.initStatus.globalAttribute;
-        config = core.clone(config || {});
-        config.left = config.left || 0;
-        config.right =
-          config.left + (config.maxWidth == null ? core._PX_ : config.maxWidth);
-        config.top = config.top || 0;
-        config.color = core.arrayToRGBA(config.color || textAttribute.text);
-        if (config.bold == null) config.bold = textAttribute.bold;
-        config.italic = config.italic || false;
-        config.align = config.align || textAttribute.align || "left";
-        config.fontSize = config.fontSize || textAttribute.textfont;
-        config.lineHeight = config.lineHeight || config.fontSize * 1.3;
-        config.defaultFont = config.font = config.font || globalAttribute.font;
-        config.time = config.time || 0;
-        config.letterSpacing =
-          config.letterSpacing == null
-            ? textAttribute.letterSpacing || 0
-            : config.letterSpacing;
+					core.doAction();
+				} else if (!core.status.event.data) {
+					cg.style.display = "none";
+					core.ui._animateUI("hide", null, () => {
+						core.doAction();
+					});
+				} else {
+					// 正在淡入淡出的话不执行
+					if (core.status.event.animateUI) return;
+					auto = false;
 
-        config.index = 0;
-        config.currcolor = config.color;
-        config.currfont = config.fontSize;
-        config.lineMargin = Math.max(
-          Math.round(config.fontSize / 4),
-          config.lineHeight - config.fontSize
-        );
-        config.topMargin = parseInt(config.lineMargin / 2);
-        config.lineMaxHeight = config.lineMargin + config.fontSize;
-        config.offsetX = 0;
-        config.offsetY = 0;
-        config.line = 0;
-        config.blocks = [];
-        config.isHD = ctx == null || ctx.canvas.hasAttribute("isHD");
+					// 打字机效果显示全部文字
+					if (core.status.event.interval != null) {
+						const data = core.clone(core.status.event.data?.current);
+						data.showAll = true;
+						data.time = 0;
+						data.text = this.text.replaceAll(/(\\(z))(\[.*?\])?/g, ""); //去除打字机暂停效果
+						data.sound = "";
+						core.insertAction(data);
+						core.doAction();
+						return;
+					} else {
+						core.stopSound(this.beforeSound);
+					}
 
-        // 创建一个新的临时画布
-        var tempCtx = document.createElement("canvas").getContext("2d");
-        if (config.isHD && ctx) {
-          core.maps._setHDCanvasSize(
-            tempCtx,
-            ctx.canvas.width,
-            ctx.canvas.height
-          );
-        } else {
-          tempCtx.canvas.width = ctx == null ? 1 : ctx.canvas.width;
-          tempCtx.canvas.height = ctx == null ? 1 : ctx.canvas.height;
-        }
+					cg.style.display = "none";
+					core.ui._animateUI("hide", null, () => {
+						core.doAction();
+					});
+				}
+			}
+		}
 
-        tempCtx.textBaseline = "top";
-        tempCtx.font = core.ui._buildFont(
-          config.fontSize,
-          config.bold,
-          config.italic,
-          config.font
-        );
-        tempCtx.fillStyle = config.color;
-        config = this._drawTextContent_draw(ctx, tempCtx, content, config);
-        return config;
-      }
-      _drawTextContent_draw(ctx, tempCtx, content, config) {
-        // Step 1: 绘制到tempCtx上,并记录下图块信息
-        while (core.ui._drawTextContent_next(tempCtx, content, config));
+		drawTextContent(ctx, content, config) {
+			//绘制多行文字并执行打字机效果
 
-        if (ctx == null) return config;
+			ctx = core.getContextByName(ctx);
+			// 设置默认配置项
+			var textAttribute =
+				core.status.textAttribute || core.initStatus.textAttribute;
+			var globalAttribute =
+				core.status.globalAttribute || core.initStatus.globalAttribute;
+			config = core.clone(config || {});
+			config.left = config.left || 0;
+			config.right =
+				config.left + (config.maxWidth == null ? core._PX_ : config.maxWidth);
+			config.top = config.top || 0;
+			config.color = core.arrayToRGBA(config.color || textAttribute.text);
+			if (config.bold == null) config.bold = textAttribute.bold;
+			config.italic = config.italic || false;
+			config.align = config.align || textAttribute.align || "left";
+			config.fontSize = config.fontSize || textAttribute.textfont;
+			config.lineHeight = config.lineHeight || config.fontSize * 1.3;
+			config.defaultFont = config.font = config.font || globalAttribute.font;
+			config.time = config.time || 0;
+			config.letterSpacing =
+				config.letterSpacing == null ?
+				textAttribute.letterSpacing || 0 :
+				config.letterSpacing;
 
-        // Step 2: 从tempCtx绘制到画布上
-        config.index = 0;
-        var _drawNext = function () {
-          if (config.index >= config.blocks.length) return false;
-          var block = config.blocks[config.index++];
-          if (block != null) {
-            // It works, why?
-            const scale = config.isHD
-              ? devicePixelRatio * core.domStyle.scale
-              : 1;
-            ctx.restore();
-            ctx.save(); //保存设置
-            if (core.domStyle.isVertical) {
-              ctx.translate(1248, 0); //重新定位右上角为基准
-              ctx.rotate(Math.PI / 2); //旋转90度
-            }
-            ctx.drawImage(
-              tempCtx.canvas,
-              block.left * scale,
-              block.top * scale,
-              block.width * scale,
-              block.height * scale,
-              config.left + block.left + block.marginLeft,
-              config.top + block.top + block.marginTop,
-              block.width,
-              block.height
-            );
-            ctx.restore();
-          }
-          return true;
-        };
+			config.index = 0;
+			config.currcolor = config.color;
+			config.currfont = config.fontSize;
+			config.lineMargin = Math.max(
+				Math.round(config.fontSize / 4),
+				config.lineHeight - config.fontSize
+			);
+			config.topMargin = parseInt(config.lineMargin / 2);
+			config.lineMaxHeight = config.lineMargin + config.fontSize;
+			config.offsetX = 0;
+			config.offsetY = 0;
+			config.line = 0;
+			config.blocks = [];
+			config.isHD = ctx == null || ctx.canvas.hasAttribute("isHD");
 
-        if (config.time == 0) {
-          while (_drawNext());
-          if (
-            (auto && !this.WindowSkin && !core.ui.cgText.sound) ||
-            (core.ui.cgText.sound && !core.musicStatus.soundStatus)
-          ) {
-            setTimeout(() => {
-              if (auto) {
-                cg.style.display = "none";
-                core.ui._animateUI("hide", null, () => {
-                  core.doAction();
-                });
-              }
-            }, core.ui.cgText.wait);
-          }
-        } else {
-          clearInterval(core.status.event.interval);
-          core.status.event.interval = setInterval(function () {
-            if (!_drawNext()) {
-              clearInterval(core.status.event.interval);
-              core.status.event.interval = null;
-              if (
-                (auto && !this.WindowSkin && !core.ui.cgText.sound) ||
-                (core.ui.cgText.sound && !core.musicStatus.soundStatus)
-              )
-                setTimeout(() => {
-                  if (auto) {
-                    cg.style.display = "none";
-                    core.ui._animateUI("hide", null, () => {
-                      core.doAction();
-                    });
-                  }
-                }, core.ui.cgText.wait);
-            }
-          }, config.time);
-        }
+			// 创建一个新的临时画布
+			var tempCtx = document.createElement("canvas").getContext("2d");
+			if (config.isHD && ctx) {
+				core.maps._setHDCanvasSize(
+					tempCtx,
+					ctx.canvas.width,
+					ctx.canvas.height
+				);
+			} else {
+				tempCtx.canvas.width = ctx == null ? 1 : ctx.canvas.width;
+				tempCtx.canvas.height = ctx == null ? 1 : ctx.canvas.height;
+			}
 
-        return config;
-      }
+			tempCtx.textBaseline = "top";
+			tempCtx.font = core.ui._buildFont(
+				config.fontSize,
+				config.bold,
+				config.italic,
+				config.font
+			);
+			tempCtx.fillStyle = config.color;
+			config = this._drawTextContent_draw(ctx, tempCtx, content, config);
+			return config;
+		}
+		_drawTextContent_draw(ctx, tempCtx, content, config) {
+			// Step 1: 绘制到tempCtx上,并记录下图块信息
+			while (core.ui._drawTextContent_next(tempCtx, content, config));
 
-      update() {
-        this.background();
-      }
-      background() {
-        const img = core.material.images.images?.[this.image];
+			if (ctx == null) return config;
 
-        if (core.domStyle.isVertical) {
-          ctx.canvas.width = 1248;
-          ctx.canvas.height = 2028;
-          ctx.save(); //保存设置
-          ctx.translate(1248, 0); //重新定位右上角为基准
-          ctx.rotate(Math.PI / 2); //旋转90度
-        } else {
-          ctx.canvas.width = 2028;
-          ctx.canvas.height = 1248;
-        }
+			// Step 2: 从tempCtx绘制到画布上
+			config.index = 0;
+			var _drawNext = function () {
+				if (config.index >= config.blocks.length) return false;
+				var block = config.blocks[config.index++];
+				if (block != null) {
+					// It works, why?
+					const scale = config.isHD ?
+						devicePixelRatio * core.domStyle.scale :
+						1;
+					ctx.restore();
+					ctx.save(); //保存设置
+					if (core.domStyle.isVertical) {
+						ctx.translate(1248, 0); //重新定位右上角为基准
+						ctx.rotate(Math.PI / 2); //旋转90度
+					}
+					ctx.drawImage(
+						tempCtx.canvas,
+						block.left * scale,
+						block.top * scale,
+						block.width * scale,
+						block.height * scale,
+						config.left + block.left + block.marginLeft,
+						config.top + block.top + block.marginTop,
+						block.width,
+						block.height
+					);
+					ctx.restore();
+				}
+				return true;
+			};
 
-        if (img) {
-          //绘制背景
-          if (this.memory) ctx.filter = "sepia(50%)";
-          ctx.drawImage(img, 0, 0, 2028, 1248);
-          ctx.filter = "none";
-        } else {
-          core.fillRect(ctx, 0, 0, 2028, 1248);
-        }
-        this.bodyList.forEach((v) => {
-          //绘制立绘
-          const body = core.material.images.images?.[v.name];
-          if (v.filter) ctx.filter = "brightness(50%)";
-          if (body) {
-            if (!v.w && !v.h && !v.scale) v.scale = 1.7;
-            if (!v.w && !v.h) {
-              ctx.drawImage(
-                body,
-                0,
-                0,
-                body.width,
-                body.height,
-                v.px,
-                1248 - body.height * v.scale,
-                body.width * v.scale,
-                body.height * v.scale
-              );
-            } else {
-              ctx.drawImage(
-                body,
-                0,
-                0,
-                body.width,
-                body.height,
-                v.px,
-                1248 - (v.h ?? body.height),
-                v.w ?? body.width,
-                v.h ?? body.height
-              );
-            }
-          }
-          ctx.filter = "none";
-        });
-        if (core.isPlaying() && !this.WindowSkin)
-          core.drawWindowSkin("winskin.webp", ctx, 30, 802, 1968, 416); //绘制对话框
-        const head = core.material.images.images?.[this.head.name];
-        if (head) {
-          //绘制头像
-          ctx.drawImage(
-            head,
-            0,
-            0,
-            head.width,
-            head.height,
-            this.head.px,
-            1248 - head.height * 2,
-            head.width * 2,
-            head.height * 2
-          );
-        }
-        if (core.isPlaying() && !this.WindowSkin) {
-          core.drawWindowSkin("winskin.webp", ctx, 1700, 1100, 192, 96);
-          core.fillBoldText1(
-            ctx,
-            "存 档",
-            1736,
-            1166,
-            "#FFFFFF",
-            "#000000",
-            6,
-            "bold 48px Verdana"
-          );
-          core.drawWindowSkin("winskin.webp", ctx, 1400, 1100, 192, 96);
-          core.fillBoldText1(
-            ctx,
-            "▶▶",
-            1456,
-            1166,
-            "#FFFFFF",
-            "#000000",
-            6,
-            "bold 48px Verdana"
-          );
-          core.drawWindowSkin("winskin.webp", ctx, 1700, 900, 192, 96);
-          let autoText = "AUTO";
-          if (auto) autoText = "STOP";
-          core.fillBoldText1(
-            ctx,
-            autoText,
-            1722,
-            966,
-            "#FFFFFF",
-            "#000000",
-            6,
-            "bold 48px Verdana"
-          );
-        }
-        if (this.name)
-          core.fillBoldText1(
-            ctx,
-            `【${this.name}】`,
-            500,
-            880,
-            "#FFFFFF",
-            "#000000",
-            6,
-            "bold 48px Verdana"
-          ); //绘制名字
-        if (
-          this.sound &&
-          core.material.sounds[this.sound] &&
-          !core.getFlag("skip", false) &&
-          core.musicStatus.soundStatus
-        ) {
-          this.beforeSound = core.playSound(this.sound, null, () => {
-            if (
-              this.sound &&
-              auto &&
-              !this.WindowSkin &&
-              core.musicStatus.soundStatus
-            ) {
-              setTimeout(() => {
-                if (auto) {
-                  cg.style.display = "none";
-                  core.ui._animateUI("hide", null, () => {
-                    core.doAction();
-                  });
-                }
-              }, core.ui.cgText.wait);
-            }
-          });
-        }
-        if (this.text && !core.getFlag("skip", false)) {
-          //绘制对话
-          this.drawTextContent(ctx, this.text, {
-            left: 500,
-            top: 950,
-            bold: true,
-            color: "#FFFFFF",
-            align: "left",
-            fontSize: 48,
-            time: this.time || 0,
-            font: "Verdana",
-            maxWidth: 1000,
-          });
-        }
+			if (config.time == 0) {
+				while (_drawNext());
 
-        ctx.restore();
-      }
-    }
-    core.ui.cgText = new cgText();
-  },
+				if (
+					(auto && !core.ui.cgText.WindowSkin && !core.ui.cgText.sound) ||
+					(core.ui.cgText.sound && !core.musicStatus.soundStatus)
+				) {
+					setTimeout(() => {
+						if (auto) {
+							cg.style.display = "none";
+							core.ui._animateUI("hide", null, () => {
+								core.doAction();
+							});
+						}
+					}, core.ui.cgText.wait);
+				}
+			} else {
+				clearInterval(core.status.event.interval);
+				core.status.event.interval = setInterval(function () {
+					if (!_drawNext()) {
+						clearInterval(core.status.event.interval);
+						core.status.event.interval = null;
+
+						if (
+							(auto && !core.ui.cgText.WindowSkin && !core.ui.cgText.sound) ||
+							(core.ui.cgText.sound && !core.musicStatus.soundStatus)
+						)
+							setTimeout(() => {
+								if (auto) {
+									cg.style.display = "none";
+									core.ui._animateUI("hide", null, () => {
+										core.doAction();
+									});
+								}
+							}, core.ui.cgText.wait);
+					}
+				}, config.time);
+			}
+
+			return config;
+		}
+
+		update() {
+			this.background()
+			if (this.log) this.logdraw(this.page);
+		}
+		logdraw(page) {
+			if (core.domStyle.isVertical) {
+				logctx.canvas.width = 1248;
+				logctx.canvas.height = 2028;
+				logctx.save(); //保存设置
+				logctx.translate(1248, 0); //重新定位右上角为基准
+				logctx.rotate(Math.PI / 2); //旋转90度
+			} else {
+				logctx.canvas.width = 2028;
+				logctx.canvas.height = 1248;
+			}
+			core.fillRect(logctx, 0, 0, 2028, 1248, "rgba(0,0,0,0.5)");
+			core.setTextAlign(logctx, "center");
+			core.fillBoldText1(
+				logctx,
+				"◀离开",
+				110,
+				100,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(66, true)
+			);
+			logctx.strokeStyle = "#FFFFFF";
+			logctx.lineWidth = 3;
+			logctx.beginPath();
+			logctx.moveTo(100, 150);
+			logctx.lineTo(1928, 150);
+
+			logctx.stroke();
+			let posy = 150;
+			const indexList = this.textList;
+			core.setTextAlign(logctx, "left");
+			for (
+				let i = (page - 1) * 6; i <= Math.min(this.index, page * 6 - 1); i++
+			) {
+				const text = core.plugin[this.textList][i][1].replaceAll(
+					/(\\(d|e|f|g|i|n|r|b|c|t|z))(\[.*?\])?/g,
+					""
+				); //取消打字机
+				const name = core.plugin[this.textList][i][0] ?
+					"【" + core.plugin[this.textList][i][0] + "】" :
+					"";
+				const sound = core.plugin[this.textList][i][2];
+				if (name) {
+					core.fillBoldText1(
+						logctx,
+						name,
+						150,
+						posy + 50,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(42, true)
+					);
+				}
+				if (sound) core.drawImage(logctx, "sound.webp", 550, posy + 30);
+				if (text.length < 30) {
+					core.fillBoldText1(
+						logctx,
+						text,
+						650,
+						posy + 50,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(42, true)
+					);
+				} else {
+					let text1 = text.slice(0, 30);
+					core.fillBoldText1(
+						logctx,
+						text1,
+						650,
+						posy + 50,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(42, true)
+					);
+					if (text.length > 60) {
+						let text2 = text.slice(30, 60);
+						let text3 = text.slice(60);
+						core.fillBoldText1(
+							logctx,
+							text2,
+							650,
+							posy + 100,
+							"#FFFFFF",
+							"#000000",
+							6,
+							core.ui._buildFont(42, true)
+						);
+						core.fillBoldText1(
+							logctx,
+							text3,
+							650,
+							posy + 150,
+							"#FFFFFF",
+							"#000000",
+							6,
+							core.ui._buildFont(42, true)
+						);
+					} else {
+						let text2 = text.slice(30);
+						core.fillBoldText1(
+							logctx,
+							text2,
+							650,
+							posy + 100,
+							"#FFFFFF",
+							"#000000",
+							6,
+							core.ui._buildFont(42, true)
+						);
+					}
+				}
+
+				logctx.strokeStyle = "#FFFFFF";
+				logctx.lineWidth = 3;
+				logctx.beginPath();
+				logctx.moveTo(100, posy);
+				logctx.lineTo(1928, posy);
+				logctx.stroke();
+
+				posy += 160;
+			}
+			logctx.beginPath();
+			logctx.moveTo(100, 1120);
+			logctx.lineTo(1928, 1120);
+			logctx.moveTo(100, 1110);
+			logctx.lineTo(1928, 1110);
+			logctx.stroke();
+			core.fillBoldText1(
+				logctx,
+				"上一页",
+				300,
+				1200,
+				page === 1 ? "#444444" : "#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(66, true)
+			);
+
+			core.fillBoldText1(
+				logctx,
+				page + "/" + this.overpage,
+				1000,
+				1200,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(66, true)
+			);
+			core.fillBoldText1(
+				logctx,
+				"下一页",
+				1500,
+				1200,
+				page === this.overpage ? "#444444" : "#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(66, true)
+			);
+		}
+
+		background() {
+			if (core.domStyle.isVertical) {
+				ctx.canvas.width = 1248;
+				ctx.canvas.height = 2028;
+				ctx.save(); //保存设置
+				ctx.translate(1248, 0); //重新定位右上角为基准
+				ctx.rotate(Math.PI / 2); //旋转90度
+			} else {
+				ctx.canvas.width = 2028;
+				ctx.canvas.height = 1248;
+			}
+
+			this.bodyList.forEach((v) => {
+				//绘制立绘
+				const body = core.material.images.images?.[v.name];
+				if (v.filter) ctx.filter = "brightness(50%)";
+				if (body) {
+					if (!v.w && !v.h && !v.scale) v.scale = 1.7;
+					if (!v.w && !v.h) {
+						ctx.drawImage(
+							body,
+							0,
+							0,
+							body.width,
+							body.height,
+							v.px,
+							1248 - body.height * v.scale,
+							body.width * v.scale,
+							body.height * v.scale
+						);
+					} else {
+						ctx.drawImage(
+							body,
+							0,
+							0,
+							body.width,
+							body.height,
+							v.px,
+							1248 - (v.h ?? body.height),
+							v.w ?? body.width,
+							v.h ?? body.height
+						);
+					}
+				}
+				ctx.filter = "none";
+			});
+			if (core.isPlaying() && !this.WindowSkin)
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx,
+					30,
+					780,
+					1968,
+					436,
+					null,
+					null,
+					null,
+					3
+				); //绘制对话框
+			const head = core.material.images.images?.[this.head.name];
+			if (head) {
+				//绘制头像
+				ctx.drawImage(
+					head,
+					0,
+					0,
+					head.width,
+					head.height,
+					this.head.px,
+					1248 - head.height * 2.2,
+					head.width * 2.2,
+					head.height * 2.2
+				);
+			}
+			if (core.isPlaying() && !this.WindowSkin) {
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx,
+					1700,
+					800,
+					192,
+					96,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx,
+					"记 录",
+					1736,
+					866,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx,
+					1700,
+					1100,
+					192,
+					96,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx,
+					"存 档",
+					1736,
+					1166,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx,
+					1700,
+					1000,
+					192,
+					96,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx,
+					"▶▶",
+					1756,
+					1066,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx,
+					1700,
+					900,
+					192,
+					96,
+					null,
+					null,
+					null,
+					3
+				);
+				let autoText = "AUTO";
+				if (auto) autoText = "STOP";
+				core.fillBoldText1(
+					ctx,
+					autoText,
+					1722,
+					966,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+			}
+			if (this.name)
+				core.fillBoldText1(
+					ctx,
+					`【${this.name}】`,
+					550,
+					880,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				); //绘制名字
+
+			if (
+				this.sound &&
+				core.sounds &&
+				!core.getFlag("skip", false) &&
+				core.musicStatus.soundStatus
+			) {
+				this.beforeSound = core.playSound(this.sound, null, () => {
+					if (
+						this.sound &&
+						auto &&
+						!this.WindowSkin &&
+						core.musicStatus.soundStatus
+					) {
+						setTimeout(() => {
+							if (auto) {
+								cg.style.display = "none";
+								core.ui._animateUI("hide", null, () => {
+									core.doAction();
+								});
+							}
+						}, this.wait);
+					}
+				});
+			}
+			if (this.text && !core.getFlag("skip", false)) {
+				//绘制对话
+				this.drawTextContent(ctx, this.text, {
+					left: 550,
+					top: 950,
+					bold: true,
+					color: "#FFFFFF",
+					align: "left",
+					fontSize: 48,
+					time: this.time || 0,
+					font: "Verdana",
+					maxWidth: 1000,
+				});
+			}
+
+			ctx.restore();
+		}
+	}
+	core.ui.cgText = new cgText();
+},
     "旁白": function () {
-    // 在此增加新插件
-    const over = document.createElement("canvas"); //over画布设置
-    over.style.position = "absolute";
-    over.style.zIndex = 310;
-    over.style.display = "none";
-    over.id = "over";
-    main.dom.gameGroup.insertAdjacentElement("afterend", over);
-    over.style.top = "50%";
-    over.style.left = "50%";
-    over.style.transform = "translate(-50%,-50%)";
-    const ctx = over.getContext("2d");
-    main.dom.over = over;
+	// 在此增加新插件
+	const over = document.createElement("canvas"); //over画布设置
+	over.style.position = "absolute";
+	over.style.zIndex = 310;
+	over.style.display = "none";
+	over.id = "over";
+	main.dom.gameGroup.insertAdjacentElement("afterend", over);
+	over.style.top = "50%";
+	over.style.left = "50%";
+	over.style.transform = "translate(-50%,-50%)";
+	const ctx1 = over.getContext("2d");
+	main.dom.over = over;
 
-    this.over = function (
-      image,
-      memory,
-      time = 100,
-      hidetime = 30,
-      sound = "",
-      textColor = "#FFFFFF",
-      boldColor = "#000000",
-      font = "bold 48px Verdana",
-      text = ""
-    ) {
-      if (!core.isPlaying()) {
-        return core.doAction();
-      }
-      const img = core.material.images.images?.[image];
-      over.style.display = "block";
-      let frame = 0;
-      let sod = 0;
-      let now = 0;
-      core.registerAnimationFrame("over", true, (timestamp) => {
-        if (timestamp - now > 1000 / 60) {
-          now = timestamp;
-          core.clearMap(ctx);
-          if (core.domStyle.isVertical) {
-            ctx.canvas.width = 416 * 3;
-            ctx.canvas.height = 676 * 3;
-            ctx.save(); //保存设置
-            ctx.translate(416 * 3, 0); //重新定位右上角为基准
-            ctx.rotate(Math.PI / 2); //旋转90度
-          } else {
-            ctx.canvas.width = 676 * 3;
-            ctx.canvas.height = 416 * 3;
-          }
-          ctx.globalAlpha = 1;
-          if (img) {
-            //绘制背景
-            if (memory) ctx.filter = "sepia(50%)";
-            ctx.drawImage(img, 0, 0, 676 * 3, 416 * 3);
-            ctx.filter = "none";
-          } else {
-            core.fillRect(ctx, 0, 0, 676 * 3, 416 * 3);
-          }
-          frame++;
-          // 绘制
-          if (frame <= hidetime)
-            core.setAlpha(ctx, 1 - (hidetime - frame) / hidetime);
-          if (frame > hidetime && frame <= hidetime + time) ctx.globalAlpha = 1;
-          if (frame > hidetime + time && frame <= hidetime * 2 + time)
-            core.setAlpha(ctx, 1 - (frame - hidetime - time) / hidetime);
-          const lisen =
-            sound &&
-            core.material.sounds[sound] &&
-            core.musicStatus.soundStatus;
-          if (frame == hidetime && lisen) {
-            sod = core.playSound(sound);
-          }
-          if (frame > hidetime * 2 + time) {
-            core.unregisterAnimationFrame("over");
-            ctx.restore();
-            over.style.display = "none";
-            core.stopSound(sod);
-            core.doAction();
-            return;
-          }
+	const temp = document.createElement("canvas")
+	temp.width = 676 * 3;
+	temp.height = 416 * 3;
 
-          core.setTextAlign(ctx, "center");
-          core.fillBoldText1(
-            ctx,
-            text,
-            1014,
-            624,
-            textColor,
-            boldColor,
-            6,
-            font
-          );
+	const ctx = temp.getContext("2d");
 
-          ctx.restore();
-        }
-      });
-    };
+	this.overupdate = function () {
+		if (core.domStyle.isVertical) {
+			ctx1.canvas.width = 416 * 3;
+			ctx1.canvas.height = 676 * 3;
+			ctx1.save(); //保存设置
+			ctx1.translate(416 * 3, 0); //重新定位右上角为基准
+			ctx1.rotate(Math.PI / 2); //旋转90度
+		} else {
+			ctx1.canvas.width = 676 * 3;
+			ctx1.canvas.height = 416 * 3;
+		}
+		core.clearMap(ctx1)
+		ctx1.drawImage(temp, 0, 0, 676 * 3, 416 * 3)
 
-    this.overlist = function (
-      image,
-      memory,
-      hidetime = 30,
-      list = [
-        {
-          text: "",
-          sound: "",
-          time: 50,
-          textColor: "#FFFFFF",
-          boldColor: "#000000",
-          font: "bold 48px Verdana",
-          frame: 0,
-        },
-      ]
-    ) {
-      if (!core.isPlaying()) {
-        return core.doAction();
-      }
-      const img = core.material.images.images?.[image];
-      over.style.display = "block";
-      let sod = 0;
-      let i = 0;
-      let now = 0;
-      core.registerAnimationFrame("overlist", true, (timestamp) => {
-        if (timestamp - now > 1000 / 60) {
-          now = timestamp;
-          core.clearMap(ctx);
-          if (core.domStyle.isVertical) {
-            ctx.canvas.width = 416 * 3;
-            ctx.canvas.height = 676 * 3;
-            ctx.save(); //保存设置
-            ctx.translate(416 * 3, 0); //重新定位右上角为基准
-            ctx.rotate(Math.PI / 2); //旋转90度
-          } else {
-            ctx.canvas.width = 676 * 3;
-            ctx.canvas.height = 416 * 3;
-          }
-          ctx.globalAlpha = 1;
-          if (img) {
-            //绘制背景
-            if (memory) ctx.filter = "sepia(50%)";
-            ctx.drawImage(img, 0, 0, 676 * 3, 416 * 3);
-            ctx.filter = "none";
-          } else {
-            core.fillRect(ctx, 0, 0, 676 * 3, 416 * 3);
-          }
-          const a = list[i];
-          const b = list[i - 1];
-          const c = list[i - 2];
-          const d = list[i - 3];
-          let ay = 624,
-            by = 624,
-            cy = 624,
-            dy = 624;
-          if (i === 0 && !list[1]) {
-            core.over(
-              image,
-              memory,
-              a.time,
-              hidetime,
-              a.sound,
-              a.textColor,
-              a.boldColor,
-              a.font,
-              a.text
-            );
-          } else {
-            const numa =
-              parseInt(a?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
-              48;
-            const numb =
-              parseInt(b?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
-              48;
-            const numc =
-              parseInt(c?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
-              48;
-            const numd =
-              parseInt(d?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
-              48;
+		ctx1.restore();
+	}
+	this.over = function (
+		image,
+		memory,
+		time = 100,
+		hidetime = 30,
+		sound = "",
+		textColor = "#FFFFFF",
+		boldColor = "#000000",
+		font = "bold 48px Verdana",
+		text = ""
+	) {
+		if (!core.isPlaying()) {
+			return core.doAction();
+		}
+		const img = core.material.images.images?.[image];
+		over.style.display = "block";
 
-            // 绘制
-            if (a) {
-              if (a.frame < hidetime / 2) {
-                a.frame++;
-                core.setAlpha(ctx, 1 - (hidetime - a.frame) / hidetime);
-                ay += ((numa * (hidetime - a.frame)) / hidetime) * 3;
-                core.setTextAlign(ctx, "center");
-                core.fillBoldText1(
-                  ctx,
-                  a.text,
-                  1014,
-                  ay,
-                  a.textColor,
-                  a.boldColor,
-                  6,
-                  a.font
-                );
-              }
-              if (a.frame === hidetime / 2) {
-                core.setAlpha(ctx, 1 - (hidetime - a.frame) / hidetime);
-                ay = 624 + ((numa * (hidetime - a.frame)) / hidetime) * 3;
-                core.setTextAlign(ctx, "center");
+		let sod = 0;
+		let start = 0;
+		core.registerAnimationFrame("over", true, (timestamp) => {
 
-                core.fillBoldText1(
-                  ctx,
-                  a.text,
-                  1014,
-                  ay,
-                  a.textColor,
-                  a.boldColor,
-                  6,
-                  a.font
-                );
-                if (!b) {
-                  a.frame++;
-                  i++;
-                }
-              }
-            }
-            if (b) {
-              if (b.frame > hidetime / 2 && b.frame <= hidetime) {
-                b.frame++;
-                core.setAlpha(ctx, 1 - (hidetime - b.frame) / hidetime);
-                by += ((numb * (hidetime - b.frame)) / hidetime) * 3;
-                core.setTextAlign(ctx, "center");
-                core.fillBoldText1(
-                  ctx,
-                  b.text,
-                  1014,
-                  by,
-                  b.textColor,
-                  b.boldColor,
-                  6,
-                  b.font
-                );
+			if (start === 0) start = timestamp;
+			core.clearMap(ctx);
 
-                core.stopSound(sod);
-              }
-              const lisenb =
-                b.sound &&
-                core.material.sounds[b.sound] &&
-                core.musicStatus.soundStatus;
-              if (b.frame && lisenb) {
-                sod = core.playSound(sound);
-              }
-              if (b.frame > hidetime && b.frame < hidetime + b.time) {
-                b.frame++;
-                ctx.globalAlpha = 1;
-                core.setTextAlign(ctx, "center");
-                core.fillBoldText1(
-                  ctx,
-                  b.text,
-                  1014,
-                  by,
-                  b.textColor,
-                  b.boldColor,
-                  6,
-                  b.font
-                );
-              }
 
-              if (b.frame == hidetime + b.time) {
-                ctx.globalAlpha = 1;
-                core.setTextAlign(ctx, "center");
-                core.fillBoldText1(
-                  ctx,
-                  b.text,
-                  1014,
-                  by,
-                  b.textColor,
-                  b.boldColor,
-                  6,
-                  b.font
-                );
-                if (a) a.frame++;
-                if (b) b.frame++;
-                if (c) c.frame++;
-                i++;
-              }
-            }
-            if (c) {
-              if (
-                c.frame > hidetime + c.time &&
-                c.frame < (hidetime * 3) / 2 + c.time
-              ) {
-                c.frame++;
-                core.setAlpha(
-                  ctx,
-                  1 - (c.frame - hidetime - c.time) / hidetime
-                );
-                cy -= ((numc * (c.frame - hidetime - c.time)) / hidetime) * 3;
-                core.setTextAlign(ctx, "center");
-                core.fillBoldText1(
-                  ctx,
-                  c.text,
-                  1014,
-                  cy,
-                  c.textColor,
-                  c.boldColor,
-                  6,
-                  c.font
-                );
-              }
-              if (c.frame === (hidetime * 3) / 2 + c.time) {
-                core.setAlpha(
-                  ctx,
-                  1 - (c.frame - hidetime - c.time) / hidetime
-                );
-                cy =
-                  624 - ((numc * (c.frame - hidetime - c.time)) / hidetime) * 3;
-                core.setTextAlign(ctx, "center");
-                core.fillBoldText1(
-                  ctx,
-                  c.text,
-                  1014,
-                  cy,
-                  c.textColor,
-                  c.boldColor,
-                  6,
-                  c.font
-                );
-                if (!b) {
-                  c.frame++;
-                  i++;
-                }
-              }
-            }
-            if (d) {
-              if (
-                d.frame > (hidetime * 3) / 2 + d.time &&
-                d.frame < hidetime * 2 + d.time
-              ) {
-                d.frame++;
-                core.setAlpha(
-                  ctx,
-                  1 - (d.frame - hidetime - d.time) / hidetime
-                );
-                dy -= ((numd * (d.frame - hidetime - d.time)) / hidetime) * 3;
-                core.setTextAlign(ctx, "center");
-                core.fillBoldText1(
-                  ctx,
-                  d.text,
-                  1014,
-                  dy,
-                  d.textColor,
-                  d.boldColor,
-                  6,
-                  d.font
-                );
-              }
-              if (d.frame == hidetime * 2 + d.time && !c) {
-                core.unregisterAnimationFrame("overlist");
-                ctx.restore();
-                over.style.display = "none";
-                core.stopSound(sod);
-                core.doAction();
-                return;
-              }
-            }
-          }
-          ctx.restore();
-        }
-      });
-    };
+			ctx.globalAlpha = 1;
+			if (img) {
+				//绘制背景
 
-    this.changebg = function (img1, memory1, img2, memory2, time, style) {
-      let globalAlpha1 = 0;
-      let globalAlpha2 = time;
-      img1 = core.material.images.images?.[img1];
-      img2 = core.material.images.images?.[img2];
-      over.style.display = "block";
-      let now = 0;
-      switch (style) {
-        case "引入":
-          core.registerAnimationFrame("bgin", true, (timestamp) => {
-            if (timestamp - now > 1000 / 60) {
-              now = timestamp;
-              core.clearMap(ctx);
-              if (core.domStyle.isVertical) {
-                ctx.canvas.width = 1248;
-                ctx.canvas.height = 676 * 3;
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-              } else {
-                ctx.canvas.width = 676 * 3;
-                ctx.canvas.height = 1248;
-              }
-              ctx.globalAlpha = globalAlpha1 / time;
-              if (img2) {
-                //绘制背景
-                if (memory2) ctx.filter = "sepia(50%)";
-                ctx.drawImage(img2, 0, 0, 676 * 3, 1248);
-                ctx.filter = "none";
-              } else {
-                core.fillRect(ctx, 0, 0, 676 * 3, 1248);
-              }
-              globalAlpha1++;
-              ctx.restore();
-              if (globalAlpha1 >= time) {
-                core.unregisterAnimationFrame("bgin");
-                over.style.display = "none";
-                core.doAction();
-              }
-            }
-          });
+				if (memory) ctx.filter = "sepia(50%)";
+				ctx.drawImage(img, 0, 0, 676 * 3, 416 * 3);
+				ctx.filter = "none";
 
-          break;
-        case "引出":
-          core.registerAnimationFrame("bgout", true, (timestamp) => {
-            if (timestamp - now > 1000 / 60) {
-              now = timestamp;
-              core.clearMap(ctx);
-              if (core.domStyle.isVertical) {
-                ctx.canvas.width = 1248;
-                ctx.canvas.height = 676 * 3;
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-              } else {
-                ctx.canvas.width = 676 * 3;
-                ctx.canvas.height = 1248;
-              }
-              ctx.globalAlpha = globalAlpha2 / time;
-              if (img1) {
-                //绘制背景
-                if (memory1) ctx.filter = "sepia(50%)";
-                ctx.drawImage(img1, 0, 0, 676 * 3, 1248);
-                ctx.filter = "none";
-              } else {
-                core.fillRect(ctx, 0, 0, 676 * 3, 1248);
-              }
-              globalAlpha2--;
-              ctx.restore();
-              if (globalAlpha2 <= 0) {
-                core.unregisterAnimationFrame("bgout");
-                over.style.display = "none";
-                core.doAction();
-              }
-            }
-          });
+			} else {
+				ctx.fillStyle = "#000000"
+				core.fillRect(ctx, 0, 0, 676 * 3, 416 * 3);
 
-          break;
-        case "场景切换":
-          core.registerAnimationFrame("changebg", true, (timestamp) => {
-            if (timestamp - now > 1000 / 60) {
-              now = timestamp;
-              core.clearMap(ctx);
-              if (core.domStyle.isVertical) {
-                ctx.canvas.width = 1248;
-                ctx.canvas.height = 676 * 3;
-                ctx.save(); //保存设置
-                ctx.translate(1248, 0); //重新定位右上角为基准
-                ctx.rotate(Math.PI / 2); //旋转90度
-              } else {
-                ctx.canvas.width = 676 * 3;
-                ctx.canvas.height = 1248;
-              }
-              ctx.globalAlpha = 1;
-              core.fillRect(ctx, 0, 0, 676 * 3, 1248);
-              ctx.globalAlpha = globalAlpha2 / time;
-              if (img1) {
-                //绘制背景
-                if (memory1) ctx.filter = "sepia(50%)";
-                ctx.drawImage(img1, 0, 0, 676 * 3, 1248);
-                ctx.filter = "none";
-              } else {
-                core.fillRect(ctx, 0, 0, 676 * 3, 1248);
-              }
-              ctx.globalAlpha = globalAlpha1 / time;
-              if (img2) {
-                //绘制背景
-                if (memory2) ctx.filter = "sepia(50%)";
-                ctx.drawImage(img2, 0, 0, 676 * 3, 1248);
-                ctx.filter = "none";
-              } else {
-                core.fillRect(ctx, 0, 0, 676 * 3, 1248);
-              }
-              globalAlpha2--;
-              globalAlpha1++;
-              ctx.restore();
-              if (globalAlpha2 <= 0 || globalAlpha1 >= time) {
-                core.unregisterAnimationFrame("changebg");
-                over.style.display = "none";
-                core.doAction();
-              }
-            }
-          });
+			}
+			let frame = Math.floor((timestamp - start) / (1000 / 60))
+			// 绘制
+			if (frame <= hidetime)
+				core.setAlpha(ctx, 1 - (hidetime - frame) / hidetime);
+			if (frame > hidetime && frame <= hidetime + time) ctx.globalAlpha = 1;
+			if (frame > hidetime + time && frame <= hidetime * 2 + time)
+				core.setAlpha(ctx, 1 - (frame - hidetime - time) / hidetime);
+			const lisen =
+				sound && core.sounds[sound] && core.musicStatus.soundStatus;
+			if (frame == hidetime && lisen) {
+				sod = core.playSound(sound);
+			}
+			if (frame > hidetime * 2 + time) {
+				core.unregisterAnimationFrame("over");
+				ctx.restore();
+				over.style.display = "none";
+				core.stopSound(sod);
+				core.doAction();
+				return;
+			}
 
-          break;
-      }
-    };
-  },
+			core.setTextAlign(ctx, "center");
+			core.fillBoldText1(
+				ctx,
+				text,
+				1014,
+				624,
+				textColor,
+				boldColor,
+				6,
+				font
+			);
+			if (core.domStyle.isVertical) {
+				ctx1.canvas.width = 416 * 3;
+				ctx1.canvas.height = 676 * 3;
+				ctx1.save(); //保存设置
+				ctx1.translate(416 * 3, 0); //重新定位右上角为基准
+				ctx1.rotate(Math.PI / 2); //旋转90度
+			} else {
+				ctx1.canvas.width = 676 * 3;
+				ctx1.canvas.height = 416 * 3;
+			}
+			core.clearMap(ctx1)
+			ctx1.drawImage(temp, 0, 0, 676 * 3, 416 * 3)
+			ctx1.restore();
+
+		});
+	};
+
+	this.overlist = function (
+		image,
+		memory,
+		hidetime = 30,
+		list = [{
+			text: "",
+			sound: "",
+			time: 50,
+			textColor: "#FFFFFF",
+			boldColor: "#000000",
+			font: "bold 48px Verdana",
+			frame: 0,
+		}, ]
+	) {
+		if (!core.isPlaying()) {
+			return core.doAction();
+		}
+		const img = core.material.images.images?.[image];
+		over.style.display = "block";
+		let sod = 0;
+		let i = 0;
+		let now = 0;
+		core.registerAnimationFrame("overlist", true, (timestamp) => {
+			if (timestamp - now > 1000 / 60) {
+				now = timestamp;
+				core.clearMap(ctx);
+
+				ctx.globalAlpha = 1;
+				if (img) {
+					//绘制背景
+					if (memory) ctx.filter = "sepia(50%)";
+
+					core.drawImage(ctx, img, 0, 0, 676 * 3, 416 * 3);
+					ctx.filter = "none";
+				} else {
+					ctx.fillStyle = "#000000"
+					core.fillRect(ctx, 0, 0, 676 * 3, 416 * 3);
+				}
+				const a = list[i];
+				const b = list[i - 1];
+				const c = list[i - 2];
+				const d = list[i - 3];
+				let ay = 624,
+					by = 624,
+					cy = 624,
+					dy = 624;
+				if (i === 0 && !list[1]) {
+					core.over(
+						image,
+						memory,
+						a.time,
+						hidetime,
+						a.sound,
+						a.textColor,
+						a.boldColor,
+						a.font,
+						a.text
+					);
+				} else {
+					const numa =
+						parseInt(a?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
+						48;
+					const numb =
+						parseInt(b?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
+						48;
+					const numc =
+						parseInt(c?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
+						48;
+					const numd =
+						parseInt(d?.font?.match(/\s*[\d.-]+[a-zA-Z%]*\s*/)?.[0].trim()) ||
+						48;
+
+					// 绘制
+					if (a) {
+						if (a.frame < hidetime / 2) {
+							a.frame++;
+							core.setAlpha(ctx, 1 - (hidetime - a.frame) / hidetime);
+							ay += ((numa * (hidetime - a.frame)) / hidetime) * 3;
+							core.setTextAlign(ctx, "center");
+							core.fillBoldText1(
+								ctx,
+								a.text,
+								1014,
+								ay,
+								a.textColor,
+								a.boldColor,
+								6,
+								a.font
+							);
+						}
+						if (a.frame === hidetime / 2) {
+							core.setAlpha(ctx, 1 - (hidetime - a.frame) / hidetime);
+							ay = 624 + ((numa * (hidetime - a.frame)) / hidetime) * 3;
+							core.setTextAlign(ctx, "center");
+
+							core.fillBoldText1(
+								ctx,
+								a.text,
+								1014,
+								ay,
+								a.textColor,
+								a.boldColor,
+								6,
+								a.font
+							);
+							if (!b) {
+								a.frame++;
+								i++;
+							}
+						}
+					}
+					if (b) {
+						if (b.frame > hidetime / 2 && b.frame <= hidetime) {
+							b.frame++;
+							core.setAlpha(ctx, 1 - (hidetime - b.frame) / hidetime);
+							by += ((numb * (hidetime - b.frame)) / hidetime) * 3;
+							core.setTextAlign(ctx, "center");
+							core.fillBoldText1(
+								ctx,
+								b.text,
+								1014,
+								by,
+								b.textColor,
+								b.boldColor,
+								6,
+								b.font
+							);
+
+							core.stopSound(sod);
+						}
+						const lisenb =
+							b.sound && core.sounds[b.sound] && core.musicStatus.soundStatus;
+						if (b.frame && lisenb) {
+							sod = core.playSound(sound);
+						}
+						if (b.frame > hidetime && b.frame < hidetime + b.time) {
+							b.frame++;
+							ctx.globalAlpha = 1;
+							core.setTextAlign(ctx, "center");
+							core.fillBoldText1(
+								ctx,
+								b.text,
+								1014,
+								by,
+								b.textColor,
+								b.boldColor,
+								6,
+								b.font
+							);
+						}
+
+						if (b.frame == hidetime + b.time) {
+							ctx.globalAlpha = 1;
+							core.setTextAlign(ctx, "center");
+							core.fillBoldText1(
+								ctx,
+								b.text,
+								1014,
+								by,
+								b.textColor,
+								b.boldColor,
+								6,
+								b.font
+							);
+							if (a) a.frame++;
+							if (b) b.frame++;
+							if (c) c.frame++;
+							i++;
+						}
+					}
+					if (c) {
+						if (
+							c.frame > hidetime + c.time &&
+							c.frame < (hidetime * 3) / 2 + c.time
+						) {
+							c.frame++;
+							core.setAlpha(
+								ctx,
+								1 - (c.frame - hidetime - c.time) / hidetime
+							);
+							cy -= ((numc * (c.frame - hidetime - c.time)) / hidetime) * 3;
+							core.setTextAlign(ctx, "center");
+							core.fillBoldText1(
+								ctx,
+								c.text,
+								1014,
+								cy,
+								c.textColor,
+								c.boldColor,
+								6,
+								c.font
+							);
+						}
+						if (c.frame === (hidetime * 3) / 2 + c.time) {
+							core.setAlpha(
+								ctx,
+								1 - (c.frame - hidetime - c.time) / hidetime
+							);
+							cy =
+								624 - ((numc * (c.frame - hidetime - c.time)) / hidetime) * 3;
+							core.setTextAlign(ctx, "center");
+							core.fillBoldText1(
+								ctx,
+								c.text,
+								1014,
+								cy,
+								c.textColor,
+								c.boldColor,
+								6,
+								c.font
+							);
+							if (!b) {
+								c.frame++;
+								i++;
+							}
+						}
+					}
+					if (d) {
+						if (
+							d.frame > (hidetime * 3) / 2 + d.time &&
+							d.frame < hidetime * 2 + d.time
+						) {
+							d.frame++;
+							core.setAlpha(
+								ctx,
+								1 - (d.frame - hidetime - d.time) / hidetime
+							);
+							dy -= ((numd * (d.frame - hidetime - d.time)) / hidetime) * 3;
+							core.setTextAlign(ctx, "center");
+							core.fillBoldText1(
+								ctx,
+								d.text,
+								1014,
+								dy,
+								d.textColor,
+								d.boldColor,
+								6,
+								d.font
+							);
+						}
+						if (d.frame == hidetime * 2 + d.time && !c) {
+							core.unregisterAnimationFrame("overlist");
+							core.clearMap(ctx)
+							core.clearMap(ctx1)
+
+							over.style.display = "none";
+							core.stopSound(sod);
+							core.doAction();
+							return;
+						}
+					}
+				}
+				if (core.domStyle.isVertical) {
+					ctx1.canvas.width = 416 * 3;
+					ctx1.canvas.height = 676 * 3;
+					ctx1.save(); //保存设置
+					ctx1.translate(416 * 3, 0); //重新定位右上角为基准
+					ctx1.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx1.canvas.width = 676 * 3;
+					ctx1.canvas.height = 416 * 3;
+				}
+				core.clearMap(ctx1)
+				ctx1.drawImage(temp, 0, 0, 676 * 3, 416 * 3)
+				ctx1.restore();
+			}
+		});
+	};
+
+	this.changebg = function (img1, memory1, img2, memory2, time, style) {
+		let frame = 0;
+		img1 = core.material.images.images?.[img1];
+		img2 = core.material.images.images?.[img2];
+		over.style.display = "block";
+		let start = 0;
+		switch (style) {
+		case "引入":
+			core.registerAnimationFrame("bgin", true, (timestamp) => {
+
+				if (start === 0) start = timestamp;
+				let frame = Math.floor((timestamp - start) / (1000 / 60))
+				core.clearMap(ctx);
+
+				ctx.globalAlpha = frame / time;
+				if (img2) {
+					//绘制背景
+					if (memory2) ctx.filter = "sepia(50%)";
+					core.drawImage(ctx, img2, 0, 0, 676 * 3, 1248);
+					ctx.filter = "none";
+				} else {
+					ctx.fillStyle = "#000000"
+					core.fillRect(ctx, 0, 0, 676 * 3, 1248);
+				}
+
+				if (core.domStyle.isVertical) {
+					ctx1.canvas.width = 416 * 3;
+					ctx1.canvas.height = 676 * 3;
+					ctx1.save(); //保存设置
+					ctx1.translate(416 * 3, 0); //重新定位右上角为基准
+					ctx1.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx1.canvas.width = 676 * 3;
+					ctx1.canvas.height = 416 * 3;
+				}
+				core.clearMap(ctx1)
+				ctx1.drawImage(temp, 0, 0, 676 * 3, 416 * 3)
+				ctx1.restore();
+				if (frame > time) {
+					core.unregisterAnimationFrame("bgin");
+
+					core.doAction();
+				}
+
+
+			});
+
+			break;
+		case "引出":
+			core.registerAnimationFrame("bgout", true, (timestamp) => {
+				if (start === 0) start = timestamp;
+				let frame = Math.floor((timestamp - start) / (1000 / 60))
+				core.clearMap(ctx);
+
+				ctx.globalAlpha = 1 - frame / time;
+				if (img1) {
+					//绘制背景
+					if (memory1) ctx.filter = "sepia(50%)";
+					core.drawImage(ctx, img1, 0, 0, 676 * 3, 1248);
+					ctx.filter = "none";
+				} else {
+					ctx.fillStyle = "#000000"
+					core.fillRect(ctx, 0, 0, 676 * 3, 1248);
+				}
+
+				if (core.domStyle.isVertical) {
+					ctx1.canvas.width = 416 * 3;
+					ctx1.canvas.height = 676 * 3;
+					ctx1.save(); //保存设置
+					ctx1.translate(416 * 3, 0); //重新定位右上角为基准
+					ctx1.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx1.canvas.width = 676 * 3;
+					ctx1.canvas.height = 416 * 3;
+				}
+				core.clearMap(ctx1)
+				ctx1.drawImage(temp, 0, 0, 676 * 3, 416 * 3)
+				ctx1.restore();
+				if (frame > time) {
+					core.unregisterAnimationFrame("bgout");
+					over.style.display = "none";
+					core.doAction();
+				}
+
+			});
+
+			break;
+		case "场景切换":
+			core.registerAnimationFrame("changebg", true, (timestamp) => {
+				if (start === 0) start = timestamp;
+				let frame = Math.floor((timestamp - start) / (1000 / 60))
+				core.clearMap(ctx);
+
+				ctx.globalAlpha = 1;
+				ctx.fillStyle = "#000000"
+				core.fillRect(ctx, 0, 0, 676 * 3, 1248);
+				ctx.globalAlpha = 1 - frame / time;
+				if (img1) {
+					//绘制背景
+					if (memory1) ctx.filter = "sepia(50%)";
+					core.drawImage(ctx, img1, 0, 0, 676 * 3, 1248);
+					ctx.filter = "none";
+				} else {
+					ctx.fillStyle = "#000000"
+					core.fillRect(ctx, 0, 0, 676 * 3, 1248);
+				}
+				ctx.globalAlpha = frame / time;
+				if (img2) {
+					//绘制背景
+					if (memory2) ctx.filter = "sepia(50%)";
+					core.drawImage(ctx, img2, 0, 0, 676 * 3, 1248);
+					ctx.filter = "none";
+				} else {
+					ctx.fillStyle = "#000000"
+					core.fillRect(ctx, 0, 0, 676 * 3, 1248);
+				}
+
+				if (core.domStyle.isVertical) {
+					ctx1.canvas.width = 416 * 3;
+					ctx1.canvas.height = 676 * 3;
+					ctx1.save(); //保存设置
+					ctx1.translate(416 * 3, 0); //重新定位右上角为基准
+					ctx1.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx1.canvas.width = 676 * 3;
+					ctx1.canvas.height = 416 * 3;
+				}
+				core.clearMap(ctx1)
+				ctx1.drawImage(temp, 0, 0, 676 * 3, 416 * 3)
+				ctx1.restore();
+				if (frame > time) {
+					core.unregisterAnimationFrame("changebg");
+
+					core.doAction();
+				}
+
+			});
+
+			break;
+
+		}
+	};
+	this.drawbackground = function (image, memory) {
+
+
+		over.style.display = "block";
+
+		//绘制背景
+		if (memory) ctx.filter = "sepia(70%)";
+		core.drawImage(ctx, image, 0, 0, 2028, 1248);
+		ctx.filter = "none";
+		if (core.domStyle.isVertical) {
+			ctx1.canvas.width = 416 * 3;
+			ctx1.canvas.height = 676 * 3;
+			ctx1.save(); //保存设置
+			ctx1.translate(416 * 3, 0); //重新定位右上角为基准
+			ctx1.rotate(Math.PI / 2); //旋转90度
+		} else {
+			ctx1.canvas.width = 676 * 3;
+			ctx1.canvas.height = 416 * 3;
+		}
+		core.clearMap(ctx1)
+		ctx1.drawImage(temp, 0, 0, 676 * 3, 416 * 3)
+		ctx1.restore();
+
+	}
+	this.clearbackground = function () {
+
+		over.style.display = "none";
+		core.doAction();
+	}
+},
     "回合制boss战": function () {
-    // 在此增加新插件
-    const boss = document.createElement("canvas"); //boss战画布设置
-    boss.style.position = "absolute";
-    boss.style.zIndex = 300;
-    boss.style.display = "none";
-    boss.id = "boss";
-    main.dom.gameGroup.insertAdjacentElement("afterend", boss);
-    boss.style.top = "50%";
-    boss.style.left = "50%";
-    boss.style.transform = "translate(-50%,-50%)";
-    const ctx = boss.getContext("2d");
-    main.dom.boss = boss;
-    boss.onclick = function (e) {
-      try {
-        e.preventDefault();
-        if (core.isPlaying()) return false;
-        const left = core.dom.gameGroup.offsetLeft;
-        const top = core.dom.gameGroup.offsetTop;
-        const px = Math.floor((e.clientX - left) / core.domStyle.scale),
-          py = Math.floor((e.clientY - top) / core.domStyle.scale);
-        core.ui.boss.onclick(px * 3, py * 3);
-      } catch (ee) {
-        main.log(ee);
-      }
-    };
-    class boss1 {
-      constructor() {
-        this.bg = "";
-      }
-      cavans() {
-        if (core.domStyle.isVertical) {
-          ctx.canvas.width = 1248;
-          ctx.canvas.height = 2028;
+	// 在此增加新插件
+	const boss = document.createElement("canvas"); //boss战画布设置
+	boss.style.position = "absolute";
+	boss.style.zIndex = 310;
+	boss.style.display = "none";
+	boss.id = "boss";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss);
+	boss.style.top = "50%";
+	boss.style.left = "50%";
+	boss.style.transform = "translate(-50%,-50%)";
+	const ctx = boss.getContext("2d");
+	main.dom.boss = boss;
+	const boss1 = document.createElement("canvas"); //boss战画布设置
+	boss1.style.position = "absolute";
+	boss1.style.zIndex = 300;
+	boss1.style.display = "none";
+	boss1.id = "boss1";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss1);
+	boss1.style.top = "50%";
+	boss1.style.left = "50%";
+	boss1.style.transform = "translate(-50%,-50%)";
+	const ctx1 = boss1.getContext("2d");
+	main.dom.boss1 = boss1;
+	const boss2 = document.createElement("canvas"); //boss战画布设置
+	boss2.style.position = "absolute";
+	boss2.style.zIndex = 301;
+	boss2.style.display = "none";
+	boss2.id = "boss2";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss2);
+	boss2.style.top = "50%";
+	boss2.style.left = "50%";
+	boss2.style.transform = "translate(-50%,-50%)";
+	main.dom.boss2 = boss2;
+	const ctx2 = boss2.getContext("2d");
+	const boss3 = document.createElement("canvas"); //boss战画布设置
+	boss3.style.position = "absolute";
+	boss3.style.zIndex = 302;
+	boss3.style.display = "none";
+	boss3.id = "boss3";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss3);
+	boss3.style.top = "50%";
+	boss3.style.left = "50%";
+	boss3.style.transform = "translate(-50%,-50%)";
+	main.dom.boss3 = boss3;
+	const ctx3 = boss3.getContext("2d");
+	const boss4 = document.createElement("canvas"); //boss战画布设置
+	boss4.style.position = "absolute";
+	boss4.style.zIndex = 303;
+	boss4.style.display = "none";
+	boss4.id = "boss4";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss4);
+	boss4.style.top = "50%";
+	boss4.style.left = "50%";
+	boss4.style.transform = "translate(-50%,-50%)";
+	const ctx4 = boss4.getContext("2d");
+	main.dom.boss4 = boss4;
+	const boss5 = document.createElement("canvas"); //boss战画布设置
+	boss5.style.position = "absolute";
+	boss5.style.zIndex = 304;
+	boss5.style.display = "none";
+	boss5.id = "boss5";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss5);
+	boss5.style.top = "50%";
+	boss5.style.left = "50%";
+	boss5.style.transform = "translate(-50%,-50%)";
+	const ctx5 = boss5.getContext("2d");
+	main.dom.boss5 = boss5;
 
-          core.setTextAlign(ctx, "center");
-        } else {
-          ctx.canvas.width = 2028;
-          ctx.canvas.height = 1248;
+	const boss6 = document.createElement("canvas"); //boss战画布设置
+	boss6.style.position = "absolute";
+	boss6.style.zIndex = 305;
+	boss6.style.display = "none";
+	boss6.id = "boss6";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss6);
+	boss6.style.top = "50%";
+	boss6.style.left = "50%";
+	boss6.style.transform = "translate(-50%,-50%)";
+	const ctx6 = boss6.getContext("2d");
+	main.dom.boss6 = boss6;
+	const boss7 = document.createElement("canvas"); //boss战画布设置
+	boss7.style.position = "absolute";
+	boss7.style.zIndex = 306;
+	boss7.style.display = "none";
+	boss7.id = "boss7";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss7);
+	boss7.style.top = "50%";
+	boss7.style.left = "50%";
+	boss7.style.transform = "translate(-50%,-50%)";
+	const ctx7 = boss7.getContext("2d");
+	main.dom.boss7 = boss7;
+	const boss8 = document.createElement("canvas"); //boss战画布设置
+	boss8.style.position = "absolute";
+	boss8.style.zIndex = 307;
+	boss8.style.display = "none";
+	boss8.id = "boss8";
+	main.dom.gameGroup.insertAdjacentElement("afterend", boss8);
+	boss8.style.top = "50%";
+	boss8.style.left = "50%";
+	boss8.style.transform = "translate(-50%,-50%)";
+	const ctx8 = boss8.getContext("2d");
+	main.dom.boss8 = boss8;
+	const { imagelighter } = core.plugin.utils;
+	const { Animation, linear, bezier, circle, hyper, trigo, power, inverseTrigo, shake, sleep } = core.plugin.animate
 
-          core.setTextAlign(ctx, "center");
-        }
-      }
-      start() {}
-      onclick(px, py) {}
-    }
-    core.ui.boss = new boss1();
-  },
+	function animateFarme(name, animate, centerX, centerY, index) {
+		var a = core.getContextByName(name);
+		if (!a) return;
+		var frame = animate.frames[index % animate.frame];
+
+		core.playSound((animate.se || {})[index % animate.frame + 1], (animate.pitch || {})[index % animate.frame + 1]);
+		var ratio = animate.ratio * 6;
+		if (core.domStyle.isVertical) {
+			a.canvas.width = 1248;
+			a.canvas.height = 2028;
+			a.save(); //保存设置
+			a.translate(1248, 0); //重新定位右上角为基准
+			a.rotate(Math.PI / 2); //旋转90度
+		} else {
+			a.canvas.width = 2028;
+			a.canvas.height = 1248;
+		}
+		frame.forEach(function (t) {
+			var image = animate.images[t.index];
+			if (!image) return;
+
+			var realWidth = image.width * ratio * t.zoom / 100;
+			var realHeight = image.height * ratio * t.zoom / 100;
+			core.setAlpha(a, t.opacity / 255);
+
+			var cx = centerX + t.x * 6,
+				cy = centerY + t.y * 6;
+
+			var ix = cx - realWidth / 2,
+				iy = cy - realHeight / 2;
+
+			var mirror = t.mirror ? 'x' : null;
+			var angle = t.angle ? -t.angle * Math.PI / 180 : null;
+			core.drawImage(a, image, ix, iy, realWidth, realHeight, null, null, null, null, angle, mirror);
+
+			core.setAlpha(a, 1);
+			a.restore()
+		})
+	}
+	let ani = []
+
+	function drawanimate(timestamp) {
+		let frametime = timestamp - core.ui.boss.time
+		core.ui.boss.time = timestamp;
+		if (!ani || ani.length == 0) return;
+
+		core.clearMap(ctx6);
+		// 更新帧
+		for (var i = 0; i < ani.length; i++) {
+			var obj = ani[i];
+			if (obj.start === 0) obj.start = core.ui.boss.time
+			obj.index = Math.floor((core.ui.boss.time - obj.start) / (1000 / 60))
+
+			if (obj.index >= obj.animate.frames.length) {
+
+				(function (callback) {
+					setTimeout(function () {
+						if (callback) callback();
+					});
+				})(obj.callback);
+			}
+		}
+		ani = ani.filter(function (obj) {
+			return obj.index < obj.animate.frames.length;
+		});
+
+		ani.forEach(function (obj) {
+
+			animateFarme(ctx6, obj.animate, obj.centerX, obj.centerY, obj.index);
+
+		});
+	}
+
+
+	function getClick() {
+		return new Promise((resolve) => {
+			function handleBossClick(e) {
+				try {
+					e.preventDefault();
+					if (!core.isPlaying()) return false;
+
+					const left = core.dom.gameGroup.offsetLeft;
+					const top = core.dom.gameGroup.offsetTop;
+					const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+						py = Math.floor((e.clientY - top) / core.domStyle.scale);
+
+					let x, y;
+					if (core.domStyle.isVertical) {
+						x = py * 3;
+						y = 1248 - px * 3;
+					} else {
+						x = px * 3;
+						y = py * 3;
+					}
+					core.ui.boss.click(x, y);
+
+					if (
+						x > 1050 &&
+						x < 1450 &&
+						y > 250 &&
+						y < 390 &&
+						!core.ui.boss.show &&
+						core.ui.boss.hasEnemy()
+					) {
+						// 移除事件监听器
+						boss.removeEventListener("click", handleBossClick);
+						resolve("普通攻击"); //有小怪时点击普通攻击
+						return; // 退出函数
+					}
+					if (
+						x > 850 &&
+						x < 1250 &&
+						y > 250 &&
+						y < 390 &&
+						!core.ui.boss.show &&
+						!core.ui.boss.hasEnemy()
+					) {
+						// 移除事件监听器
+						boss.removeEventListener("click", handleBossClick);
+						resolve("普通攻击"); //无小怪时点击普通攻击
+						return; // 退出函数
+					}
+				} catch (ee) {
+					main.log(ee);
+				}
+			}
+			boss.addEventListener("click", handleBossClick);
+		});
+	}
+
+	class Boss {
+		constructor() {
+			//绘制需要的变量
+			this.enemyfarme = 0;
+			this.bg = "bg_3512.webp";
+			this.heroImage = "tati_310101.webp";
+			this.hero = {
+				id: "hero",
+				name: "凯伊姆",
+				hp: 1000,
+				atk: 150,
+				def: 100,
+				spell: 100,
+				speed: 10,
+				mdef: 10,
+			};
+			this.boss = {
+				name: "菲奥奈",
+				id: "angel",
+				image: "tati_050143.webp",
+				hp: 1000,
+				atk: 200,
+				def: 100,
+				speed: 10,
+				mdef: 10,
+				skill: ["普通攻击", "重斩"],
+				index: 0,
+			};
+			this.onchoose = false
+			this.enemy = [];
+			this.enemy = [{
+					name: "小蝙蝠",
+					id: "bat",
+					image: "tati_020125a.webp",
+					hp: 1000,
+					atk: 100,
+					def: 20,
+					speed: 10,
+					mdef: 10,
+					skill: ["普通攻击", "重斩"],
+					index: 0,
+				},
+				{
+					name: "红蝙蝠",
+					id: "redBat",
+					image: "tati_050301.webp",
+					hp: 1000,
+					atk: 100,
+					def: 120,
+					speed: 10,
+					mdef: 10,
+					skill: ["普通攻击", "重斩"],
+					index: 0,
+				},
+				{
+					name: "大蝙蝠",
+					id: "bigBat",
+					image: "tati_120101.webp",
+					hp: 1000,
+					atk: 100,
+					def: 100,
+					speed: 10,
+					mdef: 10,
+					skill: ["普通攻击", "重斩"],
+					index: 0,
+				},
+				{
+					name: "绿色史莱姆",
+					id: "greenSlime",
+					image: "tati_340115.webp",
+					hp: 1000,
+					atk: 100,
+					def: 100,
+					speed: 10,
+					mdef: 10,
+					skill: ["普通攻击", "重斩"],
+					index: 0,
+				},
+				{
+					name: "红色史莱姆",
+					id: "redSlime",
+					image: "tati_430101.webp",
+					hp: 1000,
+					atk: 100,
+					def: 100,
+					speed: 10,
+					mdef: 10,
+					skill: ["普通攻击", "重斩"],
+					index: 0,
+				},
+				{
+					name: "黑色史莱姆",
+					id: "blackSlime",
+					image: "tati_440101.webp",
+					hp: 1000,
+					atk: 100,
+					def: 100,
+					speed: 10,
+					mdef: 10,
+					skill: ["普通攻击", "重斩"],
+					index: 0,
+				},
+			];
+			this.selection = "boss";
+			this.herobuff = [
+				{ id: "sword1", count: 1 },
+				{ id: "fly", count: 30 },
+			];
+			this.bossbuff = [
+				{ id: "fly", count: 2 },
+				{ id: "book", count: 12, hp: 100 },
+			];
+			this.enemybuff = [
+				[],
+				[],
+				[],
+				[],
+				[],
+				[]
+			];
+			this.skills = {
+				//技能列表,便于调用(可通过this.skills[name]调用)
+				菲奥奈: ["普通攻击", "重斩"],
+			};
+			this.skillShow = {
+				//技能说明
+				普通攻击: "常规攻击形式,伤害为自身攻击-对手防御",
+			};
+
+			this.turn = 0;
+			this.playingAnimate = new Set();
+			this.playerTurn = false;
+			this.show = false;
+			this.time = 0
+		}
+		buffshow(a) {
+			//buff说明(均未实装,之后改为实际需要的buff)
+			let text = "";
+			switch (a.id) {
+			case "sword1":
+				text = `\r[rgb(30,66,30)]剑气:\r造成伤害提升\r[red]${
+              10 * a.count
+            }%\r`;
+				break;
+			case "fly":
+				text = `羽翼守护:接下来\r[gold]${a.count}回合\r不受速度差值影响`;
+				break;
+			case "book":
+				text = `恢复之书:接下来\r[#000033]${a.count}回合\r,每回合行动时回复\r[green]${a.hp}\r点生命`;
+				break;
+			}
+			return text;
+		}
+		async shake(hero) {
+			const animate = new Animation()
+			const fn = () => {
+				if (hero) {
+					const img = imagelighter(
+						core.material.images.images[this.heroImage]
+					);
+					core.clearMap(ctx2);
+					if (core.domStyle.isVertical) {
+						ctx2.canvas.width = 1248;
+						ctx2.canvas.height = 2028;
+						ctx2.save(); //保存设置
+						ctx2.translate(1248, 0); //重新定位右上角为基准
+						ctx2.rotate(Math.PI / 2); //旋转90度
+					} else {
+						ctx2.canvas.width = 2028;
+						ctx2.canvas.height = 1248;
+					}
+
+					core.drawImage(ctx2, img, animate.x, 168, 750, 1080);
+					ctx2.restore();
+				} else {
+					core.clearMap(ctx3);
+					if (core.domStyle.isVertical) {
+						ctx3.canvas.width = 1248;
+						ctx3.canvas.height = 2028;
+						ctx3.save(); //保存设置
+						ctx3.translate(1248, 0); //重新定位右上角为基准
+						ctx3.rotate(Math.PI / 2); //旋转90度
+					} else {
+						ctx3.canvas.width = 2028;
+						ctx3.canvas.height = 1248;
+					}
+					if (this.selection === "boss" || this.selection === "") {
+						const img = imagelighter(
+							core.material.images.images[this.boss.image]
+						);
+						core.drawImage(ctx3, img, 1400 + animate.x, 168, 750, 1080);
+					} else {
+						const img = imagelighter(
+							core.material.images.images[
+								this.enemy[this.selection].image
+							]
+						);
+						core.drawImage(ctx3, img, 1400 + animate.x, 168, 750, 1080);
+					}
+
+					ctx3.restore();
+				}
+			}
+			animate.ticker.add(fn)
+			animate.getTime = () => Date.now() / 100
+			animate.mode(shake(30, linear()), true).time(5).shake(1, 0)
+			await animate.all();
+			animate.ticker.destroy()
+			this.drawhero();
+			this.drawboss();
+		}
+
+		popDamage(damage, onhero) {
+
+			let color = "#FFFFFF";
+			if (typeof damage === "number") {
+				color = damage < 0 ? "#22FF44" : "lightcoral";
+			}
+			let posx = onhero ? 300 : 1800;
+			let start = 0
+			const speed = 9;
+			return new Promise((resolve) => {
+				core.registerAnimationFrame("popDamageonboss", true, (temptime) => {
+
+					if (start === 0) start = temptime
+
+					let farme = Math.floor((temptime - start) / (1000 / 60))
+					if (core.domStyle.isVertical) {
+						ctx.canvas.width = 1248;
+						ctx.canvas.height = 2028;
+						ctx.save(); //保存设置
+						ctx.translate(1248, 0); //重新定位右上角为基准
+						ctx.rotate(Math.PI / 2); //旋转90度
+					} else {
+						ctx.canvas.width = 2028;
+						ctx.canvas.height = 1248;
+					}
+					core.clearMap(ctx);
+					core.setTextAlign(ctx, "center");
+					core.fillBoldText1(
+						ctx,
+						damage,
+						posx,
+						800 - speed * farme,
+						color,
+						"#000000",
+						6,
+						"bold 72px Arial"
+					);
+
+					ctx.restore();
+					if (farme > 30) {
+						core.unregisterAnimationFrame("popDamageonboss");
+
+						core.clearMap(ctx);
+						resolve();
+					}
+
+				});
+			});
+		}
+		async skill(sk, a, b) {
+			//a为发起方属性,sk为技能名
+			let damage = 0;
+			switch (
+				sk //所有技能效果及动画写在这里
+			) {
+			case "普通攻击":
+				damage = Math.max(a.atk - b.def, 0); //基础伤害
+				if (a.id === 'hero') damage = Math.floor((damage * a.speed) / b.speed); //勇者攻击时伤害根据速度比值伤害加成
+				b.hp -= damage; //承受伤害
+				if (b.id === "hero")
+					core.status.hero.statistics.battleDamage += damage; //数据统计记录伤害
+				if (damage === 0) damage = "抵抗";
+
+				switch (
+					a.id //根据id选取不同的特效
+				) {
+				case "hero":
+					this.popDamage(damage, false);
+					if (damage > 0 && typeof damage !== "string") this.shake(false);
+					await this.playanimate("jianji2", 1750, 770); //播放动画jianji2
+					break;
+				case "angel":
+					this.popDamage(damage, true);
+					if (damage > 0 && typeof damage !== "string") this.shake(true);
+					await this.playanimate("sword", 300, 770); //播放动画sword
+
+					break;
+				case "bat":
+					this.popDamage(damage, true);
+					if (damage > 0 && typeof damage !== "string") this.shake(true);
+					await this.playanimate("sword", 300, 770); //播放动画sword
+
+					break;
+				case "redBat":
+					this.popDamage(damage, true);
+					if (damage > 0 && typeof damage !== "string") this.shake(true);
+					await this.playanimate("hand", 300, 770); //播放动画hand
+					break;
+				case "bigBat":
+					this.popDamage(damage, true);
+					if (damage > 0 && typeof damage !== "string") this.shake(true);
+					await this.playanimate("jianji", 300, 770); //播放动画jianji
+					break;
+				case "greenSlime":
+					this.popDamage(damage, true);
+					if (damage > 0 && typeof damage !== "string") this.shake(true);
+					await this.playanimate("thunder", 300, 770); //播放动画thunder
+					break;
+				case "redSlime":
+					this.popDamage(damage, true);
+					if (damage > 0 && typeof damage !== "string") this.shake(true);
+					await this.playanimate("yongchang", 300, 770); //播放动画yongchang
+					break;
+				case "blackSlime":
+					this.popDamage(damage, true);
+					if (damage > 0 && typeof damage !== "string") this.shake(true);
+					await this.playanimate("zone", 300, 770); //播放动画zone
+					break;
+				}
+				break; //下面写其余技能
+			}
+			await sleep(500); //等待1000ms
+		}
+
+		click(px, py) {
+			//点击效果
+			const makeBox = ([x, y], [w, h]) => {
+				return [
+					[x, y],
+					[x + w, y + h],
+				];
+			};
+			const inRect = ([x, y], [
+				[sx, sy],
+				[dx, dy]
+			]) => {
+				return sx <= x && x <= dx && sy <= y && y <= dy;
+			};
+			const pos = [px, py];
+			if (main.replayChecking || core.isReplaying()) return;
+			const enemyStatusBox = makeBox([50, 50], [1900, 200]),
+				heroStatusBox = makeBox([600, 920], [900, 300]),
+				bossBox = makeBox([800, 300], [100, 150]),
+				enemyBox = makeBox([700, 500], [300, 300]),
+				imageBox = makeBox([1500, 250], [550, 1000]);
+			if (this.show) {
+				//清除展示画面
+				this.show = !this.show;
+				core.clearMap(ctx8);
+			} else {
+				if (
+					inRect(pos, enemyStatusBox) &&
+					this.selection !== "" &&
+					this.selection !== "boss" &&
+					this.hasEnemy()
+				) {
+					//绘制怪物详情
+					this.show = !this.show;
+					this.moreShow(this.selection);
+				} else if (
+					inRect(pos, enemyStatusBox) &&
+					(!this.hasEnemy() || this.selection === "boss")
+				) {
+					//绘制boss详情
+					this.show = !this.show;
+					this.moreShow("boss");
+				} else if (inRect(pos, heroStatusBox)) {
+					//绘制勇士详情
+					this.show = !this.show;
+					this.moreShow("hero");
+				} else if (inRect(pos, bossBox) && this.hasEnemy()) {
+					//切换selection为boss
+					this.selection = "boss";
+					this.update();
+				} else if (inRect(pos, enemyBox) && this.hasEnemy()) {
+					//切换selection为enemy
+					const symbol =
+						Math.floor((px - 700) / 100) + Math.floor((py - 500) / 150) * 3;
+
+					if (this.enemy[symbol] && this.enemy[symbol].hp > 0) {
+						this.selection = symbol;
+						this.update();
+					}
+				}
+			}
+		}
+		drawchoose() {
+			boss7.style.display = "block";
+			if (core.domStyle.isVertical) {
+				ctx7.canvas.width = 1248;
+				ctx7.canvas.height = 2028;
+				ctx7.save(); //保存设置
+				ctx7.translate(1248, 0); //重新定位右上角为基准
+				ctx7.rotate(Math.PI / 2); //旋转90度
+			} else {
+				ctx7.canvas.width = 2028;
+				ctx7.canvas.height = 1248;
+			}
+
+			core.clearMap(ctx7);
+			if (this.hasEnemy()) {
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx7,
+					1050,
+					250,
+					400,
+					660,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx7,
+					"普通攻击",
+					1120,
+					350,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 64px Verdana"
+				);
+				core.drawLine(ctx7, 1070, 390, 1430, 390, "#FFFFFF", 6);
+			} else {
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx7,
+					850,
+					250,
+					400,
+					660,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx7,
+					"普通攻击",
+					920,
+					350,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 64px Verdana"
+				);
+				core.drawLine(ctx7, 850, 390, 1250, 390, "#FFFFFF", 6);
+			}
+			ctx7.restore();
+		}
+		moreShow(select) {
+			if (core.domStyle.isVertical) {
+				ctx8.canvas.width = 1248;
+				ctx8.canvas.height = 2028;
+				ctx8.save(); //保存设置
+				ctx8.translate(1248, 0); //重新定位右上角为基准
+				ctx8.rotate(Math.PI / 2); //旋转90度
+			} else {
+				ctx8.canvas.width = 2028;
+				ctx8.canvas.height = 1248;
+			}
+			core.clearMap(ctx8);
+			core.fillRect(ctx8, 100, 100, 1800, 1000, "rgba(0,0,0,0.7)");
+			core.drawWindowSkin(
+				"winskin.webp",
+				ctx8,
+				100,
+				100,
+				1800,
+				1000,
+				null,
+				null,
+				null,
+				3
+			);
+			let posy = 200;
+			switch (select) {
+			case "hero":
+				//勇士技能/buff
+				core.fillBoldText1(
+					ctx8,
+					this.hero.name,
+					1000,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				posy += 100;
+				if (this.herobuff.length === 0) {
+					core.fillBoldText1(
+						ctx8,
+						"当前无特殊状态",
+						200,
+						posy,
+						"#FFFFFF",
+						"#000000",
+						6,
+						"bold 48px Verdana"
+					);
+					posy += 100;
+				} else {
+					this.herobuff.forEach((v) => {
+						core.drawTextContent(ctx8, this.buffshow(v), {
+							left: 200,
+							top: posy,
+							bold: true,
+							color: "#FFFFFF",
+							align: "left",
+							fontSize: 48,
+							time: 0,
+							font: "Verdana",
+							maxWidth: 1600,
+						});
+						posy += 100;
+					});
+				}
+				posy += 50;
+				core.setTextAlign(ctx8, "center")
+				core.fillBoldText1(
+					ctx8,
+					"主角所造成的伤害将提升主角速度与怪物敌人速度值之比",
+					1000,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				posy += 100;
+				core.fillBoldText1(
+					ctx8,
+					"技能说明",
+					1000,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				posy += 100;
+				core.fillBoldText1(
+					ctx8,
+					"普通攻击:" + this.skillShow["普通攻击"],
+					200,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				break;
+			case "boss":
+				core.fillBoldText1(
+					ctx8,
+					this.boss.name,
+					1000,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				posy += 100;
+				//boss技能/buff
+				if (this.bossbuff.length === 0) {
+					core.fillBoldText1(
+						ctx8,
+						"当前无特殊状态",
+						200,
+						posy,
+						"#FFFFFF",
+						"#000000",
+						6,
+						"bold 48px Verdana"
+					);
+					posy += 100;
+				} else {
+					this.bossbuff.forEach((v) => {
+						core.drawTextContent(ctx8, this.buffshow(v), {
+							left: 200,
+							top: posy,
+							bold: true,
+							color: "#FFFFFF",
+							align: "left",
+							fontSize: 48,
+							time: 0,
+							font: "Verdana",
+							maxWidth: 1600,
+						});
+						posy += 100;
+					});
+				}
+				posy += 50;
+				core.fillBoldText1(
+					ctx8,
+					"当前技能",
+					1000,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				posy += 100;
+				core.fillBoldText1(
+					ctx8,
+					this.boss.skill[this.boss.index] +
+					this.skillShow[this.boss.skill[this.boss.index]],
+					200,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				break;
+			default:
+				const enemy = this.enemy[select];
+				const enemybuff = this.enemybuff[select];
+				core.fillBoldText1(
+					ctx8,
+					enemy.name,
+					1000,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				posy += 100;
+				if (enemybuff.length === 0) {
+					core.fillBoldText1(
+						ctx8,
+						"当前无特殊状态",
+						200,
+						posy,
+						"#FFFFFF",
+						"#000000",
+						6,
+						"bold 48px Verdana"
+					);
+					posy += 100;
+				} else {
+					enemybuff.forEach((v) => {
+						core.drawTextContent(ctx8, this.buffshow(v), {
+							left: 200,
+							top: posy,
+							bold: true,
+							color: "#FFFFFF",
+							align: "left",
+							fontSize: 48,
+							time: 0,
+							font: "Verdana",
+							maxWidth: 1600,
+						});
+						posy += 100;
+					});
+				}
+				posy += 50;
+				core.fillBoldText1(
+					ctx8,
+					"当前技能",
+					1000,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				posy += 100;
+				core.fillBoldText1(
+					ctx8,
+					enemy.skill[enemy.index] +
+					this.skillShow[enemy.skill[enemy.index]],
+					200,
+					posy,
+					"#FFFFFF",
+					"#000000",
+					6,
+					"bold 48px Verdana"
+				);
+				break;
+			}
+			ctx8.restore();
+		}
+		hasEnemy() {
+			let enemy = false;
+			this.enemy.forEach((v) => {
+				if (v.hp > 0) enemy = true;
+			});
+			return enemy;
+		}
+		async bossStart() {
+			boss6.style.display = "block";
+
+			this.time = 0
+			core.registerAnimationFrame("animate-boss", true, drawanimate)
+			boss.style.display = "block";
+			this.selection = "boss";
+			this.playerturn = false;
+			this.show = false;
+			this.turn = 0;
+
+			core.lockControl();
+			await this.blackBg();
+			this.moveboss();
+			await this.movehero();
+			await this.moveStatus();
+			this.update();
+			this.fight();
+		}
+		async bossEnd() {
+			hero.hp = this.hero.hp;
+			this.selection = "boss";
+			await this.close();
+			core.unregisterAnimationFrame("animate-boss")
+			core.unlockControl();
+			core.updateStatusBar();
+			if (hero.hp <= 0) {
+				hero.hp = 0;
+				core.events.lose("BOSS战失败");
+			}
+		}
+		async fight() {
+			await this.drawturn();
+			this.selection = "boss"
+			this.update();
+			const fightList = [];
+			fightList.push(["hero", this.hero.speed]);
+			if (this.boss.hp > 0) fightList.push(["boss", this.boss.speed]);
+			this.enemy.forEach((v, i) => {
+				if (v.id && v.hp > 0) fightList.push([i, v.speed]);
+			});
+			fightList.sort((a, b) => b[1] - a[1]);
+			let damage;
+			for (const v of fightList) {
+				switch (v[0]) {
+				case "hero":
+					this.onchoose = true
+					this.drawchoose();
+					const skill = await getClick();
+					this.onchoose = false
+					core.clearMap(ctx7)
+					let select = this.boss;
+					if (this.selection !== "" && this.selection !== "boss")
+						select = this.enemy[this.selection];
+					this.skill(skill, this.hero, select);
+
+					break;
+				case "boss":
+					if (this.boss.hp > 0) {
+						this.selection = "boss";
+						this.update();
+						await sleep(500); //等待500ms
+						//这里写boss技能的效果
+						this.skill(
+							this.boss.skill[this.boss.index],
+							this.boss,
+							this.hero
+						);
+
+						this.boss.index++;
+						if (this.boss.index >= this.boss.skill.length)
+							this.boss.index = 0;
+					}
+					break;
+				default:
+					const enemy = this.enemy[v[0]];
+					if (enemy.hp > 0) {
+						this.selection = v[0];
+						this.update();
+						await sleep(500); //等待500ms
+						this.skill(enemy.skill[enemy.index], enemy, this.hero);
+						enemy.index++;
+						if (enemy.index >= enemy.skill.length) enemy.index = 0;
+					}
+					break;
+				}
+				await sleep(1000);
+				this.selection = "";
+				this.update();
+			}
+			let end = true;
+			if (this.boss.hp > 0) end = false;
+			this.enemy.forEach((v) => {
+				if (v.hp > 0) end = false;
+			});
+
+			if (this.hero.hp <= 0) end = true;
+			if (end) {
+				this.bossEnd();
+			} else {
+				this.fight();
+			}
+		}
+		async drawturn() {
+			const animate = new Animation();
+			boss8.style.display = "block";
+			this.turn++
+			const fn = () => {
+				core.clearMap(ctx);
+				if (core.domStyle.isVertical) {
+					ctx.canvas.width = 1248;
+					ctx.canvas.height = 2028;
+					ctx.save(); //保存设置
+					ctx.translate(1248, 0); //重新定位右上角为基准
+					ctx.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx.canvas.width = 2028;
+					ctx.canvas.height = 1248;
+				}
+				core.fillRect(
+					ctx,
+					0,
+					624 - animate.y,
+					2028,
+					animate.y * 2,
+					"rgba(0,0,0,0.7)"
+				);
+
+				core.setTextAlign(ctx, "center");
+				core.fillBoldText1(
+					ctx,
+					"回合 " + this.turn,
+					animate.x - 300,
+					644,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(96, true)
+				);
+				ctx.restore()
+			}
+			animate.ticker.add(fn)
+
+			animate.mode(hyper('sin', 'out')).time(1000).move(1314, 200)
+			await animate.all()
+			await sleep(500)
+			animate.mode(hyper('sin', 'in')).time(1000).move(2800, 0)
+			await animate.all()
+			core.clearMap(ctx);
+			animate.ticker.destroy()
+
+		}
+
+		drawenemy() {
+			let block,
+				time = 0;
+
+			boss5.style.display = "block";
+			core.registerAnimationFrame("enemyanimate", true, (temptime) => {
+
+				if (temptime - time > 1000 / 60) {
+					time = temptime;
+					this.enemyfarme += 1;
+					let animate = Math.floor(this.enemyfarme / 10),
+						posx = 700,
+						posy = 500;
+					core.clearMap(ctx5);
+					if (core.domStyle.isVertical) {
+						ctx5.canvas.width = 1248;
+						ctx5.canvas.height = 2028;
+						ctx5.save(); //保存设置
+						ctx5.translate(1248, 0); //重新定位右上角为基准
+						ctx5.rotate(Math.PI / 2); //旋转90度
+					} else {
+						ctx5.canvas.width = 2028;
+						ctx5.canvas.height = 1248;
+					}
+
+					if (this.hasEnemy()) {
+						core.drawWindowSkin(
+							"winskin.webp",
+							ctx5,
+							650,
+							250,
+							400,
+							660,
+							null,
+							null,
+							null,
+							3
+						);
+						if (this.selection === "boss")
+							core.strokeRect(ctx5, 800, 300, 100, 150, "yellow", 6);
+						const bossBlock = core.getBlockInfo(this.boss.id);
+						core.drawImage(
+							ctx5,
+							bossBlock.image,
+							32 * (animate % 4),
+							bossBlock.posY * 48,
+							32,
+							48,
+							800,
+							300,
+							96,
+							144
+						);
+						core.drawImage(
+							ctx5,
+							"hero.webp",
+							32 * (animate % 4),
+							144,
+							32,
+							48,
+							800,
+							750,
+							96,
+							144
+						);
+						for (let i = 0; i < this.enemy.length; i++) {
+							if (this.enemy[i].id && this.enemy[i].hp > 0) {
+								block = core.getBlockInfo(this.enemy[i].id);
+							} else {
+								posx += 100;
+								if (i === 2) {
+									posx = 700;
+									posy += 150;
+								}
+								continue;
+							}
+
+							if (block.cls === "enemys") {
+								core.drawImage(
+									ctx5,
+									block.image,
+									32 * (animate % 2),
+									block.posY * 32,
+									32,
+									32,
+									posx,
+									posy,
+									96,
+									96
+								);
+								if (this.selection === i)
+									core.strokeRect(ctx5, posx, posy, 100, 100, "yellow", 6);
+							} else {
+								core.drawImage(
+									ctx5,
+									block.image,
+									32 * (animate % 4),
+									block.posY * 48,
+									32,
+									48,
+									posx,
+									posy,
+									96,
+									144
+								);
+								if (this.selection === i)
+									core.strokeRect(ctx5, posx, posy, 100, 150, "yellow", 6);
+							}
+
+							posx += 100;
+							if (i === 2) {
+								posx = 700;
+								posy += 150;
+							}
+						}
+					}
+
+					ctx5.restore();
+				}
+			});
+		}
+		drawhero() {
+			core.clearMap(ctx2);
+			if (core.domStyle.isVertical) {
+				ctx2.canvas.width = 1248;
+				ctx2.canvas.height = 2028;
+				ctx2.save(); //保存设置
+				ctx2.translate(1248, 0); //重新定位右上角为基准
+				ctx2.rotate(Math.PI / 2); //旋转90度
+			} else {
+				ctx2.canvas.width = 2028;
+				ctx2.canvas.height = 1248;
+			}
+			core.drawImage(ctx2, this.heroImage, 0, 168, 750, 1080);
+			ctx2.restore();
+		}
+		async movehero() {
+			const animate = new Animation();
+			boss2.style.display = "block";
+			const fn = () => {
+				core.clearMap(ctx2);
+				if (core.domStyle.isVertical) {
+					ctx2.canvas.width = 1248;
+					ctx2.canvas.height = 2028;
+					ctx2.save(); //保存设置
+					ctx2.translate(1248, 0); //重新定位右上角为基准
+					ctx2.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx2.canvas.width = 2028;
+					ctx2.canvas.height = 1248;
+				}
+				core.drawImage(ctx2, this.heroImage, animate.x - 200, 168, 750, 1080);
+				ctx2.restore();
+			}
+			animate.ticker.add(fn)
+
+			animate.mode(hyper('sin', 'out')).time(1000).move(200, 0)
+			await animate.all()
+
+			core.clearMap(ctx2);
+			animate.ticker.destroy()
+			this.drawhero()
+
+		}
+		update() {
+			core.drawImage(ctx1, this.bg, 0, 0, 2028, 1248);
+			this.drawboss();
+			this.drawhero();
+			this.drawStatus();
+			let dodraw = false;
+			for (let i = 0; i < this.enemy.length; i++) {
+				if (this.enemy[i].id && this.enemy[i].hp > 0) {
+					dodraw = true;
+				}
+			}
+			if (dodraw === true) this.drawenemy();
+			if (this.onchoose) this.drawchoose()
+		}
+		async playanimate(name, x, y, callback) {
+			return new Promise(res => {
+				name = core.getMappedName(name);
+
+				// 正在播放录像:不显示动画
+				if (core.isReplaying() || !core.material.animates[name] || x == null || y == null) {
+					if (callback) callback();
+					return Promise.resolve(-1);
+				}
+
+				// 开始绘制
+				var animate = core.material.animates[name],
+					centerX = x,
+					centerY = y
+
+				animate.se = animate.se || {};
+				if (typeof animate.se == 'string') animate.se = { 1: animate.se };
+
+				var id = setTimeout(null);
+				ani.push({
+					"name": name,
+					"id": id,
+					"animate": animate,
+					"centerX": centerX,
+					"centerY": centerY,
+					"start": 0,
+					"index": 0,
+					"callback": () => {
+						if (callback) callback()
+						res()
+					}
+				});
+
+			})
+
+
+		}
+
+		drawboss() {
+			core.clearMap(ctx3);
+			if (core.domStyle.isVertical) {
+				ctx3.canvas.width = 1248;
+				ctx3.canvas.height = 2028;
+				ctx3.save(); //保存设置
+				ctx3.translate(1248, 0); //重新定位右上角为基准
+				ctx3.rotate(Math.PI / 2); //旋转90度
+			} else {
+				ctx3.canvas.width = 2028;
+				ctx3.canvas.height = 1248;
+			}
+			if (this.selection === "boss" || this.selection === "") {
+				core.drawImage(ctx3, this.boss.image, 1400, 168, 750, 1080);
+			} else {
+				core.drawImage(
+					ctx3,
+					this.enemy[this.selection].image,
+					1400,
+					168,
+					750,
+					1080
+				);
+			}
+
+			ctx3.restore();
+		}
+		async moveboss() {
+			const animate = new Animation();
+			boss3.style.display = "block";
+			const fn = () => {
+				core.clearMap(ctx3);
+				if (core.domStyle.isVertical) {
+					ctx3.canvas.width = 1248;
+					ctx3.canvas.height = 2028;
+					ctx3.save(); //保存设置
+					ctx3.translate(1248, 0); //重新定位右上角为基准
+					ctx3.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx3.canvas.width = 2028;
+					ctx3.canvas.height = 1248;
+				}
+				core.drawImage(ctx3, this.boss.image, 1600 - animate.x, 168, 750, 1080);
+
+				ctx3.restore();
+			}
+			animate.ticker.add(fn)
+
+			animate.mode(hyper('sin', 'out')).time(1000).move(200, 0)
+			await animate.all()
+
+			core.clearMap(ctx3);
+			animate.ticker.destroy()
+			this.drawboss()
+
+		}
+		drawStatus() {
+			core.clearMap(ctx4);
+			if (core.domStyle.isVertical) {
+				ctx4.canvas.width = 1248;
+				ctx4.canvas.height = 2028;
+				ctx4.save(); //保存设置
+				ctx4.translate(1248, 0); //重新定位右上角为基准
+				ctx4.rotate(Math.PI / 2); //旋转90度
+			} else {
+				ctx4.canvas.width = 2028;
+				ctx4.canvas.height = 1248;
+			}
+			core.drawWindowSkin(
+				"winskin.webp",
+				ctx4,
+				600,
+				920,
+				900,
+				300,
+				null,
+				null,
+				null,
+				3
+			);
+			core.fillBoldText1(
+				ctx4,
+				hero.name,
+				630,
+				1000,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			core.fillBoldText1(
+				ctx4,
+				"状态",
+				880,
+				1000,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			core.fillBoldText1(
+				ctx4,
+				"生命 " + this.hero.hp,
+				630,
+				1070,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			core.fillBoldText1(
+				ctx4,
+				"攻击 " + this.hero.atk,
+				630,
+				1120,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			core.fillBoldText1(
+				ctx4,
+				"防御 " + this.hero.def,
+				630,
+				1170,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			core.fillBoldText1(
+				ctx4,
+				"法强 " + this.hero.spell,
+				1080,
+				1070,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			core.fillBoldText1(
+				ctx4,
+				"法抗 " + this.hero.mdef + "%",
+				1080,
+				1120,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			core.fillBoldText1(
+				ctx4,
+				"速度 " + this.hero.speed,
+				1080,
+				1170,
+				"#FFFFFF",
+				"#000000",
+				6,
+				core.ui._buildFont(44, true)
+			);
+			let posx = 980;
+			this.herobuff.forEach((v) => {
+				if (v) {
+					core.drawIcon(ctx4, v.id, posx, 950, 64, 64);
+					core.setTextAlign(ctx4, "right");
+					core.fillBoldText1(
+						ctx4,
+						v.count,
+						posx + 50,
+						1000,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(24, true)
+					);
+					core.setTextAlign(ctx4, "left");
+					posx += 80;
+				}
+			});
+
+			if (this.selection === "boss" || !this.hasEnemy()) {
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx4,
+					50,
+					50,
+					1900,
+					200,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx4,
+					this.boss.name,
+					100,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"状态",
+					500,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"当前技能",
+					1400,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				const bosstext = this.boss.skill[this.boss.index];
+				core.fillBoldText1(
+					ctx4,
+					bosstext,
+					1600,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"生命 " + this.boss.hp,
+					100,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"攻击 " + this.boss.atk,
+					500,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"防御 " + this.boss.def,
+					900,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"法抗 " + this.boss.mdef + "%",
+					1300,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"速度 " + this.boss.speed,
+					1700,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				let posx = 600;
+				this.bossbuff.forEach((v) => {
+					if (v) {
+						core.drawIcon(ctx4, v.id, posx, 80, 64, 64);
+						core.setTextAlign(ctx4, "right");
+						core.fillBoldText1(
+							ctx4,
+							v.count,
+							posx + 50,
+							130,
+							"#FFFFFF",
+							"#000000",
+							6,
+							core.ui._buildFont(24, true)
+						);
+						core.setTextAlign(ctx4, "left");
+						posx += 80;
+					}
+				});
+			} else if (this.selection === "") {} else {
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx4,
+					50,
+					50,
+					1900,
+					200,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx4,
+					this.enemy[this.selection].name,
+					100,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"状态",
+					500,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"当前技能",
+					1400,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				const enemytext =
+					this.enemy[this.selection].skill[this.enemy[this.selection].index];
+				core.fillBoldText1(
+					ctx4,
+					enemytext,
+					1600,
+					120,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"生命 " + this.enemy[this.selection].hp,
+					100,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"攻击 " + this.enemy[this.selection].atk,
+					500,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"防御 " + this.enemy[this.selection].def,
+					900,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"法抗 " + this.enemy[this.selection].mdef + "%",
+					1300,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"速度 " + this.enemy[this.selection].speed,
+					1700,
+					220,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				let posx = 600;
+				this.enemybuff[this.selection].forEach((v) => {
+					if (v) {
+						core.drawIcon(ctx4, v.id, posx, 80, 64, 64);
+						core.setTextAlign(ctx4, "right");
+						core.fillBoldText1(
+							ctx4,
+							v.count,
+							posx + 50,
+							130,
+							"#FFFFFF",
+							"#000000",
+							6,
+							core.ui._buildFont(24, true)
+						);
+						core.setTextAlign(ctx4, "left");
+						posx += 80;
+					}
+				});
+			}
+			ctx4.restore();
+		}
+		async moveStatus() {
+			const animate = new Animation();
+			boss4.style.display = "block";
+			const fn = () => {
+				core.clearMap(ctx4);
+				if (core.domStyle.isVertical) {
+					ctx4.canvas.width = 1248;
+					ctx4.canvas.height = 2028;
+					ctx4.save(); //保存设置
+					ctx4.translate(1248, 0); //重新定位右上角为基准
+					ctx4.rotate(Math.PI / 2); //旋转90度
+				} else {
+					ctx4.canvas.width = 2028;
+					ctx4.canvas.height = 1248;
+				}
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx4,
+					600,
+					1320 - animate.y,
+					900,
+					300,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillBoldText1(
+					ctx4,
+					hero.name,
+					630,
+					1400 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"状态",
+					880,
+					1400 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"生命 " + this.hero.hp,
+					630,
+					1470 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"攻击 " + this.hero.atk,
+					630,
+					1520 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"防御 " + this.hero.def,
+					630,
+					1570 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"法强 " + this.hero.spell,
+					1080,
+					1470 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"法抗 " + this.hero.mdef + "%",
+					1080,
+					1520 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				core.fillBoldText1(
+					ctx4,
+					"速度 " + this.hero.speed,
+					1080,
+					1570 - animate.y,
+					"#FFFFFF",
+					"#000000",
+					6,
+					core.ui._buildFont(44, true)
+				);
+				let posx = 980;
+				this.herobuff.forEach((v) => {
+					if (v) {
+						core.drawIcon(ctx4, v.id, posx, 1350 - animate.y, 64, 64);
+						core.setTextAlign(ctx4, "right");
+						core.fillBoldText1(
+							ctx4,
+							v.count,
+							posx + 50,
+							1400 - animate.y,
+							"#FFFFFF",
+							"#000000",
+							6,
+							core.ui._buildFont(24, true)
+						);
+						core.setTextAlign(ctx4, "left");
+						posx += 80;
+					}
+				});
+				if (this.selection === "boss") {
+					core.drawWindowSkin(
+						"winskin.webp",
+						ctx4,
+						50,
+						-150 + animate.y / 2,
+						1900,
+						200,
+						null,
+						null,
+						null,
+						3
+					);
+					core.fillBoldText1(
+						ctx4,
+						this.boss.name,
+						100,
+						-80 + animate.y / 2,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"状态",
+						500,
+						animate.y / 2 - 80,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"生命 " + this.boss.hp,
+						100,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"攻击 " + this.boss.atk,
+						500,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"防御 " + this.boss.def,
+						900,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"法抗 " + this.boss.mdef + "%",
+						1300,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"速度 " + this.boss.speed,
+						1700,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					let posx = 600;
+					this.bossbuff.forEach((v) => {
+						if (v) {
+							core.drawIcon(ctx4, v.id, posx, animate.y / 2 - 120, 64, 64);
+							core.setTextAlign(ctx4, "right");
+							core.fillBoldText1(
+								ctx4,
+								v.count,
+								posx + 50,
+								animate.y / 2 - 70,
+								"#FFFFFF",
+								"#000000",
+								6,
+								core.ui._buildFont(24, true)
+							);
+							core.setTextAlign(ctx4, "left");
+							posx += 80;
+						}
+					});
+				} else if (this.selection === "") {} else {
+					core.drawWindowSkin(
+						"winskin.webp",
+						ctx4,
+						50,
+						animate.y / 2 - 150,
+						1900,
+						200,
+						null,
+						null,
+						null,
+						3
+					);
+					core.fillBoldText1(
+						ctx4,
+						this.enemy[this.selection].name,
+						100,
+						animate.y / 2 - 80,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"状态",
+						500,
+						animate.y / 2 - 80,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"生命 " + this.enemy[this.selection].hp,
+						100,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"攻击 " + this.enemy[this.selection].atk,
+						500,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"防御 " + this.enemy[this.selection].def,
+						900,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"法抗 " + this.enemy[this.selection].mdef + "%",
+						1300,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					core.fillBoldText1(
+						ctx4,
+						"速度 " + this.enemy[this.selection].speed,
+						1700,
+						animate.y / 2 + 20,
+						"#FFFFFF",
+						"#000000",
+						6,
+						core.ui._buildFont(44, true)
+					);
+					let posx = 600;
+					this.enemybuff[this.selection].forEach((v) => {
+						if (v) {
+							core.drawIcon(ctx4, v.id, posx, animate.y / 2 - 120, 64, 64);
+							core.setTextAlign(ctx4, "right");
+							core.fillBoldText1(
+								ctx4,
+								v.count,
+								posx + 50,
+								animate.y / 2 - 70,
+								"#FFFFFF",
+								"#000000",
+								6,
+								core.ui._buildFont(24, true)
+							);
+							core.setTextAlign(ctx4, "left");
+							posx += 80;
+						}
+					});
+				}
+				ctx4.restore();
+			}
+			animate.ticker.add(fn)
+
+			animate.mode(hyper('sin', 'out')).time(1000).move(0, 400)
+			await animate.all()
+
+			core.clearMap(ctx3);
+			animate.ticker.destroy()
+			this.drawStatus();
+
+		}
+		close() {
+			let globalAlpha = 0,
+				time = 0;
+			return new Promise((resolve) => {
+				core.registerAnimationFrame("closeblack", true, (temptime) => {
+					if (temptime - time > 1000 / 60) {
+						time = temptime;
+						core.clearMap(ctx);
+						if (core.domStyle.isVertical) {
+							ctx.canvas.width = 1248;
+							ctx.canvas.height = 2028;
+							ctx.save(); //保存设置
+							ctx.translate(1248, 0); //重新定位右上角为基准
+							ctx.rotate(Math.PI / 2); //旋转90度
+						} else {
+							ctx.canvas.width = 2028;
+							ctx.canvas.height = 1248;
+						}
+
+						ctx.globalAlpha = globalAlpha;
+
+						core.fillRect(ctx, 0, 0, 2028, 1248);
+						globalAlpha += 1 / 30;
+						ctx.restore();
+						if (globalAlpha > 1) {
+							time = 0;
+							globalAlpha = 1;
+							core.unregisterAnimationFrame("closeblack");
+							core.unregisterAnimationFrame("enemyanimate");
+							boss1.style.display = "none";
+							boss2.style.display = "none";
+							boss3.style.display = "none";
+							boss4.style.display = "none";
+							boss5.style.display = "none";
+							boss6.style.display = "none";
+							boss7.style.display = "none";
+							boss8.style.display = "none";
+
+							core.registerAnimationFrame("closeblack2", true, (temptime) => {
+								if (temptime - time > 1000 / 60) {
+									time = temptime;
+									core.clearMap(ctx);
+									if (core.domStyle.isVertical) {
+										ctx.canvas.width = 1248;
+										ctx.canvas.height = 2028;
+										ctx.save(); //保存设置
+										ctx.translate(1248, 0); //重新定位右上角为基准
+										ctx.rotate(Math.PI / 2); //旋转90度
+									} else {
+										ctx.canvas.width = 2028;
+										ctx.canvas.height = 1248;
+									}
+									ctx.globalAlpha = globalAlpha;
+									core.fillRect(ctx, 0, 0, 2028, 1248, "#000000");
+									ctx.restore();
+									globalAlpha -= 1 / 30;
+									if (globalAlpha < 0) {
+										core.unregisterAnimationFrame("closeblack2");
+
+										boss.style.display = "none";
+
+										resolve();
+									}
+								}
+							});
+						}
+					}
+				});
+			});
+		}
+
+		blackBg() {
+			let globalAlpha = 0,
+				time = 0,
+				img = core.material.images.images[this.bg];
+			boss1.style.display = "block";
+			return new Promise((resolve) => {
+				core.registerAnimationFrame("bossblack", true, (temptime) => {
+					if (temptime - time > 1000 / 60) {
+						time = temptime;
+						core.clearMap(ctx1);
+						if (core.domStyle.isVertical) {
+							ctx1.canvas.width = 1248;
+							ctx1.canvas.height = 2028;
+							ctx1.save(); //保存设置
+							ctx1.translate(1248, 0); //重新定位右上角为基准
+							ctx1.rotate(Math.PI / 2); //旋转90度
+						} else {
+							ctx1.canvas.width = 2028;
+							ctx1.canvas.height = 1248;
+						}
+
+						ctx1.globalAlpha = globalAlpha;
+
+						core.fillRect(ctx1, 0, 0, 2028, 1248);
+						globalAlpha += 1 / 30;
+						ctx1.restore();
+						if (globalAlpha > 1) {
+							time = 0;
+							globalAlpha = 0;
+							core.unregisterAnimationFrame("bossblack");
+							core.registerAnimationFrame("bossBg", true, (temptime) => {
+								if (temptime - time > 1000 / 60) {
+									time = temptime;
+									core.clearMap(ctx1);
+									if (core.domStyle.isVertical) {
+										ctx1.canvas.width = 1248;
+										ctx1.canvas.height = 2028;
+										ctx1.save(); //保存设置
+										ctx1.translate(1248, 0); //重新定位右上角为基准
+										ctx1.rotate(Math.PI / 2); //旋转90度
+									} else {
+										ctx1.canvas.width = 2028;
+										ctx1.canvas.height = 1248;
+									}
+
+									ctx1.globalAlpha = 1;
+									core.fillRect(ctx1, 0, 0, 2028, 1248);
+									ctx1.globalAlpha = globalAlpha;
+									if (img) ctx1.drawImage(img, 0, 0, 2028, 1248);
+									ctx1.restore();
+									globalAlpha += 1 / 30;
+									if (globalAlpha > 1) {
+										core.unregisterAnimationFrame("bossBg");
+										resolve();
+									}
+								}
+							});
+						}
+					}
+				});
+			});
+		}
+	}
+	core.ui.boss = new Boss();
+},
     "剧情视频引用": function () {
     // 在此增加新插件
     let a;
@@ -16070,7 +19498,6 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
         core.doAction();
         clearTimeout(a);
         core.playBgm(bgm);
-        core.resumeBgm();
       }, 2000); //这里设置定时器,定义长按500毫秒触发长按事件,时间可以自己改,个人感觉500毫秒非常合适
       return false;
     }
@@ -16243,1063 +19670,1286 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
     };
   },
     "帧动画/图片叠拼": function () {
+	// 在此增加新插件
+	this.animationDrawable = function (
+		allFarme,
+		color,
+		globalAlpha,
+		imageList,
+		soundList
+	) {
+		if (!core.isPlaying()) {
+			return core.doAction();
+		}
+		const over = main.dom.over;
+		const ctx = over.getContext("2d");
+		over.style.display = "block";
+
+
+		let now = 0;
+		let start = 0
+		core.registerAnimationFrame(
+
+			"animationDrawable",
+			true,
+			function (timestamp) {
+
+				let frametime = timestamp - now
+				now = timestamp;
+				if (!imageList || imageList.length == 0) return;
+				if (start === 0) start = now
+				now = timestamp;
+				let farme = Math.floor((now - start) / (1000 / 60))
+				if (farme > allFarme) {
+					core.unregisterAnimationFrame("animationDrawable");
+					core.doAction();
+					return
+				}
+				if (core.domStyle.isVertical) {
+					over.width = 1248;
+					over.height = 2028;
+					ctx.save(); //保存设置
+					ctx.translate(1248, 0); //重新定位右上角为基准
+					ctx.rotate(Math.PI / 2); //旋转90度
+				} else {
+					over.width = 2028;
+					over.height = 1248;
+				}
+
+				core.clearMap(ctx)
+				ctx.globalAlpha = (globalAlpha ?? 100) / 100;
+				core.fillRect(ctx, 0, 0, 2028, 1248, color);
+
+				imageList.forEach(function (one) {
+					if (
+						farme >= (one.beforefarme ?? 0) &&
+						farme <= (one.afterfarme ?? allFarme)
+					) {
+						const img = core.material.images.images?.[one.image];
+						if (img) {
+							const gla = one.globalAlpha ?? 100;
+							const agla = one.aglobalAlpha ?? gla,
+								beforefarme = one.beforefarme ?? 0;
+							const afterfarme = one.afterfarme ?? allFarme;
+
+							ctx.globalAlpha =
+								(gla +
+									((agla - gla) * (farme - beforefarme)) /
+									(afterfarme - beforefarme || 1)) /
+								100;
+
+							const cx =
+								(one.cx ?? 0) +
+								(((one.acx ?? 0) - (one.cx ?? 0)) *
+									(farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								cy =
+								(one.cy ?? 0) +
+								(((one.acy ?? 0) - (one.cy ?? 0)) *
+									(farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								cw =
+								(one.cw ?? img.width) +
+								(((one.acw ?? img.width) - (one.cw ?? img.width)) *
+									(farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								ch =
+								(one.ch ?? img.height) +
+								(((one.acw ?? img.height) - (one.cw ?? img.height)) *
+									(farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								x =
+								(one.x ?? 0) +
+								(((one.ax ?? 0) - (one.x ?? 0)) * (farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								y =
+								(one.y ?? 0) +
+								(((one.ay ?? 0) - (one.y ?? 0)) * (farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								w =
+								(one.w ?? 2028) +
+								(((one.aw ?? 2028) - (one.w ?? 2028)) *
+									(farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								h =
+								(one.h ?? 1248) +
+								(((one.ah ?? 1248) - (one.h ?? 1248)) *
+									(farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								angle =
+								(one.angle ?? 0) * Math.PI / 180 +
+								(((one.aangle ?? 0) * Math.PI / 180 - (one.angle ?? 0) * Math.PI / 180) *
+									(farme - beforefarme)) /
+								(afterfarme - beforefarme || 1);
+							core.drawImage(ctx, img, cx, cy, cw, ch, x, y, w, h, angle);
+						}
+					}
+				});
+				soundList.forEach(function (one) {
+					const lisen =
+						one.sound &&
+						core.sounds[one.sound] &&
+						core.musicStatus.soundStatus;
+					if (farme >= one.startfarme && lisen && !one.start) {
+						one.start = true
+						if (one.stopbefore) core.stopSound();
+						core.playSound(one.sound);
+					}
+				});
+				ctx.globalAlpha = 1;
+				ctx.restore();
+
+
+
+			}
+		);
+	};
+},
+    "musicMode": function () {
     // 在此增加新插件
-    this.animationDrawable = function (
-      allFarme,
-      color,
-      globalAlpha,
-      imageList,
-      soundList
-    ) {
-      if (!core.isPlaying()) {
-        return core.doAction();
+    const music = document.createElement("canvas");
+    music.style.position = "absolute";
+    music.style.zIndex = 300;
+    music.style.display = "none";
+    music.id = "music";
+    main.dom.gameGroup.insertAdjacentElement("afterend", music);
+    music.style.top = "50%";
+    music.style.left = "50%";
+    music.style.transform = "translate(-50%,-50%)";
+    const ctx = music.getContext("2d");
+    main.dom.music = music;
+
+    const audio = core.plugin.audioSystem.bgmController;
+
+    let page = 0; //初始页面
+
+    let isvolume = false;
+
+    function shuffle(arr) {
+      let n = arr.length,
+        random;
+      while (n) {
+        random = (Math.random() * n--) >>> 0;
+        [arr[n], arr[random]] = [arr[random], arr[n]];
       }
-      const over = main.dom.over;
-      const ctx = over.getContext("2d");
-      over.style.display = "block";
+      return arr;
+    }
+    music.addEventListener("mousedown", function (e) {
+      e.stopPropagation();
+      const left = core.dom.gameGroup.offsetLeft;
+      const top = core.dom.gameGroup.offsetTop;
+      const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+        py = Math.floor((e.clientY - top) / core.domStyle.scale);
+      core.ui.music.mousedown(px * 3, py * 3);
+    });
+    music.addEventListener("mousemove", function (e) {
+      e.stopPropagation();
+      const left = core.dom.gameGroup.offsetLeft;
+      const top = core.dom.gameGroup.offsetTop;
+      const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+        py = Math.floor((e.clientY - top) / core.domStyle.scale);
+      core.ui.music.mousemove(px * 3, py * 3);
+    });
+    music.addEventListener("mouseup", function (e) {
+      e.stopPropagation();
 
-      let farme = 0;
-      let now = 0;
-      core.registerAnimationFrame(
-        "animationDrawable",
-        true,
-        function (timestamp) {
-          if (timestamp - now > 1000 / 60) {
-            now = timestamp;
-            if (core.domStyle.isVertical) {
-              over.width = 1248;
-              over.height = 2028;
-              ctx.save(); //保存设置
-              ctx.translate(1248, 0); //重新定位右上角为基准
-              ctx.rotate(Math.PI / 2); //旋转90度
+      isvolume = false;
+    });
+    music.addEventListener("mouseleave", function (e) {
+      e.stopPropagation();
+
+      isvolume = false;
+    });
+    music.addEventListener("touchstart", function (e) {
+      e.preventDefault();
+      const left = core.dom.gameGroup.offsetLeft;
+      const top = core.dom.gameGroup.offsetTop;
+      const px = Math.floor(
+          (e.touches[0].clientX - left) / core.domStyle.scale
+        ),
+        py = Math.floor((e.touches[0].clientY - top) / core.domStyle.scale);
+      core.ui.music.mousedown(px * 3, py * 3);
+    });
+    music.addEventListener("touchmove", function (e) {
+      e.stopPropagation();
+      const left = core.dom.gameGroup.offsetLeft;
+      const top = core.dom.gameGroup.offsetTop;
+      const px = Math.floor(
+          (e.touches[0].clientX - left) / core.domStyle.scale
+        ),
+        py = Math.floor((e.touches[0].clientY - top) / core.domStyle.scale);
+      core.ui.music.mousemove(px * 3, py * 3);
+    });
+    music.addEventListener("touchend", function (e) {
+      e.stopPropagation();
+
+      isvolume = false;
+    });
+    music.addEventListener("touchcancel", function (e) {
+      e.stopPropagation();
+
+      isvolume = false;
+    });
+
+    class musicclass {
+      constructor() {
+        this.musics = ["theme.mp3"];
+        //music列表
+        //需全塔属性注册并保存在bgms文件夹,每个数组为显示的一页内容
+        this.musicMx = [
+          [
+            "Crawler.opus",
+            "Blood_Stain.opus",
+            "Blind_Alley.opus",
+            "Halbmond.opus",
+          ],
+          ["theme.mp3", "op.opus", "Asphodelus_Ceui.opus", "ed.opus"],
+        ];
+        //音乐别名(将在播放器内显示的音乐名,music列表内的都要有对应歌名)
+        this.musicname = {
+          "Asphodelus_Ceui.opus": "Asphodelus (Full.ver)",
+          "Blind_Alley.opus": "Blind Alley",
+          "Crawler.opus": "Crawler",
+          "op.opus": "Asphodelus",
+          "theme.mp3": "One of Episodes",
+          "ed.opus": "親愛なる世界へ",
+          "Blood_Stain.opus": "Blood Stain",
+          "Halbmond.opus": "Halbmond",
+        };
+        this.selection = [0, 0];
+        this.stop = false;
+        this.type = "xunhuan";
+        this.randomList = [];
+        this.random = 0;
+      }
+
+      //更新
+      update() {
+        this.background();
+        this.drawUI();
+      }
+      background() {
+        //画布大小设置
+        if (core.domStyle.isVertical) {
+          music.width = 1248;
+          music.height = 2028;
+        } else {
+          music.width = 2028;
+          music.height = 1248;
+        }
+      }
+
+      mousedown(px, py) {
+        //鼠标按下时
+
+        const makeBox = ([x, y], [w, h]) => {
+          return [
+            [x, y],
+            [x + w, y + h],
+          ];
+        };
+        const inRect = ([x, y], [[sx, sy], [dx, dy]]) => {
+          return sx <= x && x <= dx && sy <= y && y <= dy;
+        };
+        const pos = [px, py];
+        const backbox = makeBox([15, 35], [210, 90]);
+        if (inRect(pos, backbox)) {
+          //离开按钮是一致的,其余的记区分横竖屏
+          music.style.display = "none";
+          core.clearMap(ctx);
+
+          core.unregisterAnimationFrame("music");
+          core.restart();
+
+          return;
+        }
+        if (core.domStyle.isVertical) {
+          //竖屏
+
+          const pageupbox = makeBox([100, 1230], [200, 100]);
+          const pagedownbox = makeBox([950, 1230], [200, 100]);
+          const musicbox = makeBox(
+            [100, 200],
+            [1048, this.musicMx[page].length * 100]
+          );
+          const beforebox = makeBox([120, 1620], [100, 100]);
+          const afterbox = makeBox([780, 1620], [100, 100]);
+          const playbox = makeBox([420, 1580], [200, 200]);
+          const typebox = makeBox([1040, 1600], [120, 120]);
+
+          const volumebox = makeBox([250, 1940], [1050, 20]);
+          if (inRect(pos, pageupbox)) {
+            if (page !== 0) page -= 1;
+            return;
+          }
+          if (inRect(pos, pagedownbox)) {
+            if (page !== this.musicMx.length - 1) page += 1;
+            return;
+          }
+          if (inRect(pos, playbox)) {
+            if (this.stop) {
+              this.stop = !this.stop;
+
+              core.resumeBgm();
             } else {
-              over.width = 2028;
-              over.height = 1248;
+              this.stop = !this.stop;
+              core.pauseBgm();
             }
+            return;
+          }
+          if (inRect(pos, beforebox)) {
+            this.stop = false;
+            switch (this.type) {
+              case "danqu":
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
 
-            ctx.globalAlpha = (globalAlpha ?? 100) / 100;
-            core.fillRect(ctx, 0, 0, 2028, 1248, color);
+                page = this.selection[0];
 
-            imageList.forEach(function (one) {
-              if (
-                farme >= (one.beforefarme ?? 0) &&
-                farme <= (one.afterfarme ?? allFarme)
-              ) {
-                const img = core.material.images.images?.[one.image];
-                if (img) {
-                  const gla = one.globalAlpha ?? 100;
-                  const agla = one.aglobalAlpha ?? gla,
-                    beforefarme = one.beforefarme ?? 0;
-                  const afterfarme = one.afterfarme ?? allFarme;
-
-                  ctx.globalAlpha =
-                    (gla +
-                      ((agla - gla) * (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1)) /
-                    100;
-
-                  const cx =
-                      (one.cx ?? 0) +
-                      (((one.acx ?? 0) - (one.cx ?? 0)) *
-                        (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    cy =
-                      (one.cy ?? 0) +
-                      (((one.acy ?? 0) - (one.cy ?? 0)) *
-                        (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    cw =
-                      (one.cw ?? img.width) +
-                      (((one.acw ?? img.width) - (one.cw ?? img.width)) *
-                        (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    ch =
-                      (one.ch ?? img.height) +
-                      (((one.acw ?? img.height) - (one.cw ?? img.height)) *
-                        (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    x =
-                      (one.x ?? 0) +
-                      (((one.ax ?? 0) - (one.x ?? 0)) * (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    y =
-                      (one.y ?? 0) +
-                      (((one.ay ?? 0) - (one.y ?? 0)) * (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    w =
-                      (one.w ?? 2028) +
-                      (((one.aw ?? 2028) - (one.w ?? 2028)) *
-                        (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    h =
-                      (one.h ?? 1248) +
-                      (((one.aw ?? 1248) - (one.w ?? 1248)) *
-                        (farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1);
-                  ctx.drawImage(img, cx, cy, cw, ch, x, y, w, h);
+                break;
+              case "xunhuan":
+                for (;;) {
+                  if (this.selection[1] === 0) {
+                    if (this.selection[0] === 0) {
+                      this.selection[0] = this.musicMx.length - 1;
+                      this.selection[1] =
+                        this.musicMx[this.selection[0]].length - 1;
+                    } else {
+                      this.selection[0] -= 1;
+                      this.selection[1] =
+                        this.musicMx[this.selection[0]].length - 1;
+                    }
+                  } else {
+                    this.selection[1] -= 1;
+                  }
+                  this.random = this.randomList.indexOf(
+                    this.musicMx[this.selection[0]][this.selection[1]]
+                  );
+                  page = this.selection[0];
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
                 }
-              }
-            });
-            soundList.forEach(function (one) {
-              const lisen =
-                one.sound &&
-                core.material.sounds[one.sound] &&
-                core.musicStatus.soundStatus;
-              if (farme == one.startfarme && lisen) {
-                if (one.stopbefore) core.stopSound();
-                core.playSound(one.sound);
-              }
-            });
-            farme++;
-            ctx.globalAlpha = 1;
-            ctx.restore();
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
 
-            if (farme > allFarme) {
-              core.unregisterAnimationFrame("animationDrawable");
-              over.style.display = "none";
-              core.doAction();
+                break;
+              case "suiji":
+                for (;;) {
+                  if (this.random > 0) {
+                    this.random -= 1;
+                  } else {
+                    this.random = this.randomList.length - 1;
+                  }
+                  this.selection[0] = this.musicMx.findIndex((v) =>
+                    v.includes(this.randomList[this.random])
+                  );
+                  this.selection[1] = this.musicMx[this.selection[0]].indexOf(
+                    this.randomList[this.random]
+                  );
+
+                  page = this.selection[0];
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
+                }
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                break;
+            }
+            return;
+          }
+          if (inRect(pos, afterbox)) {
+            this.stop = false;
+            switch (this.type) {
+              case "danqu":
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                page = this.selection[0];
+                break;
+              case "xunhuan":
+                for (;;) {
+                  if (
+                    this.selection[1] ===
+                    this.musicMx[this.selection[0]].length - 1
+                  ) {
+                    if (this.selection[0] === this.musicMx.length - 1) {
+                      this.selection[0] = 0;
+                      this.selection[1] = 0;
+                    } else {
+                      this.selection[0] += 1;
+                      this.selection[1] = 0;
+                    }
+                  } else {
+                    this.selection[1] += 1;
+                  }
+                  this.random = this.randomList.indexOf(
+                    this.musicMx[this.selection[0]][this.selection[1]]
+                  );
+                  page = this.selection[0];
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
+                }
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                break;
+              case "suiji":
+                for (;;) {
+                  if (this.random < this.randomList.length - 1) {
+                    this.random += 1;
+                  } else {
+                    this.random = 0;
+                  }
+                  this.selection[0] = this.musicMx.findIndex((v) =>
+                    v.includes(this.randomList[this.random])
+                  );
+                  this.selection[1] = this.musicMx[this.selection[0]].indexOf(
+                    this.randomList[this.random]
+                  );
+
+                  page = this.selection[0];
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
+                }
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                break;
+            }
+            return;
+          }
+          if (inRect(pos, typebox)) {
+            switch (this.type) {
+              case "danqu":
+                this.type = "xunhuan";
+                break;
+              case "xunhuan":
+                this.type = "suiji";
+                break;
+              case "suiji":
+                this.type = "danqu";
+                break;
+            }
+            return;
+          }
+          if (inRect(pos, musicbox)) {
+            const index = Math.floor((py - 200) / 100);
+            if (page !== this.selection[0] || index !== this.selection[1]) {
+              if (
+                this.musics.includes(this.musicMx[page][index]) ||
+                page !== this.musicMx.length - 1
+              ) {
+                this.selection[0] = page;
+
+                this.selection[1] = index;
+                this.randomList.indexOf(
+                  this.musicMx[this.selection[0]][this.selection[1]]
+                );
+
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                this.stop = false;
+              }
+            } else {
+              if (this.stop) {
+                this.stop = !this.stop;
+                core.resumeBgm();
+              } else {
+                this.stop = !this.stop;
+                core.pauseBgm();
+              }
+            }
+            return;
+          }
+
+          if (inRect(pos, volumebox)) {
+            const time = Math.min(Math.max((px - 250) / 800, 0), 1);
+            audio.setVolume(time);
+            core.plugin.audioSystem.soundPlayer.setVolume(time);
+            isvolume = true;
+          }
+        } else {
+          //横屏
+          const pageupbox = makeBox([1050, 1100], [200, 100]);
+          const pagedownbox = makeBox([1550, 1100], [200, 100]);
+          const musicbox = makeBox(
+            [900, 100],
+            [1000, this.musicMx[page].length * 100]
+          );
+          const beforebox = makeBox([60, 620], [100, 100]);
+          const afterbox = makeBox([450, 620], [100, 100]);
+          const playbox = makeBox([200, 570], [200, 200]);
+          const typebox = makeBox([620, 600], [120, 120]);
+
+          const volumebox = makeBox([100, 990], [600, 20]);
+          if (inRect(pos, pageupbox)) {
+            if (page !== 0) page -= 1;
+            return;
+          }
+          if (inRect(pos, pagedownbox)) {
+            if (page !== this.musicMx.length - 1) page += 1;
+            return;
+          }
+          if (inRect(pos, playbox)) {
+            if (this.stop) {
+              this.stop = !this.stop;
+              core.resumeBgm();
+            } else {
+              this.stop = !this.stop;
+              core.pauseBgm();
+            }
+            return;
+          }
+          if (inRect(pos, beforebox)) {
+            this.stop = false;
+            switch (this.type) {
+              case "danqu":
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                page = this.selection[0];
+
+                break;
+              case "xunhuan":
+                for (;;) {
+                  if (this.selection[1] === 0) {
+                    if (this.selection[0] === 0) {
+                      this.selection[0] = this.musicMx.length - 1;
+                      this.selection[1] =
+                        this.musicMx[this.selection[0]].length - 1;
+                    } else {
+                      this.selection[0] -= 1;
+                      this.selection[1] =
+                        this.musicMx[this.selection[0]].length - 1;
+                    }
+                  } else {
+                    this.selection[1] -= 1;
+                  }
+                  this.random = this.randomList.indexOf(
+                    this.musicMx[this.selection[0]][this.selection[1]]
+                  );
+                  page = this.selection[0];
+
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
+                }
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                break;
+              case "suiji":
+                for (;;) {
+                  if (this.random > 0) {
+                    this.random -= 1;
+                  } else {
+                    this.random = this.randomList.length - 1;
+                  }
+                  this.selection[0] = this.musicMx.findIndex((v) =>
+                    v.includes(this.randomList[this.random])
+                  );
+                  this.selection[1] = this.musicMx[this.selection[0]].indexOf(
+                    this.randomList[this.random]
+                  );
+
+                  page = this.selection[0];
+
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
+                }
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                break;
+            }
+            return;
+          }
+          if (inRect(pos, afterbox)) {
+            this.stop = false;
+            switch (this.type) {
+              case "danqu":
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                page = this.selection[0];
+                break;
+              case "xunhuan":
+                for (;;) {
+                  if (
+                    this.selection[1] ===
+                    this.musicMx[this.selection[0]].length - 1
+                  ) {
+                    if (this.selection[0] === this.musicMx.length - 1) {
+                      this.selection[0] = 0;
+                      this.selection[1] = 0;
+                    } else {
+                      this.selection[0] += 1;
+                      this.selection[1] = 0;
+                    }
+                  } else {
+                    this.selection[1] += 1;
+                  }
+                  this.randomList.findIndex(
+                    (v) =>
+                      v === this.musicMx[this.selection[0]][this.selection[1]]
+                  );
+                  page = this.selection[0];
+
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
+                }
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                break;
+              case "suiji":
+                for (;;) {
+                  if (this.random < this.randomList.length - 1) {
+                    this.random += 1;
+                  } else {
+                    this.random = 0;
+                  }
+                  this.selection[0] = this.musicMx.findIndex((v) =>
+                    v.includes(this.randomList[this.random])
+                  );
+                  this.selection[1] = this.musicMx[this.selection[0]].indexOf(
+                    main.core.ui.music.randomList[main.core.ui.music.random]
+                  );
+
+                  page = this.selection[0];
+                  if (
+                    this.musics.includes(
+                      this.musicMx[this.selection[0]][this.selection[1]]
+                    ) ||
+                    page !== this.musicMx.length - 1
+                  )
+                    break;
+                }
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+
+                break;
+            }
+            return;
+          }
+          if (inRect(pos, typebox)) {
+            switch (this.type) {
+              case "danqu":
+                this.type = "xunhuan";
+                break;
+              case "xunhuan":
+                this.type = "suiji";
+                break;
+              case "suiji":
+                this.type = "danqu";
+                break;
+            }
+            return;
+          }
+          if (inRect(pos, musicbox)) {
+            const index = Math.floor((py - 100) / 100);
+            if (page !== this.selection[0] || index !== this.selection[1]) {
+              if (
+                this.musics.includes(this.musicMx[page][index]) ||
+                page !== this.musicMx.length - 1
+              ) {
+                this.selection[0] = page;
+                this.selection[1] = index;
+                this.randomList.indexOf(
+                  (v) =>
+                    v === this.musicMx[this.selection[0]][this.selection[1]]
+                );
+
+                core.playBgm(
+                  main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                    main.core.ui.music.selection[1]
+                  ],
+                  0
+                );
+                this.stop = false;
+              }
+            } else {
+              if (this.stop) {
+                this.stop = !this.stop;
+                core.resumeBgm();
+              } else {
+                this.stop = !this.stop;
+                core.pauseBgm();
+              }
+            }
+            return;
+          }
+
+          if (inRect(pos, volumebox)) {
+            const time = Math.min(Math.max((px - 100) / 600, 0), 1);
+            audio.setVolume(time);
+            core.plugin.audioSystem.soundPlayer.setVolume(time);
+            isvolume = true;
+          }
+        }
+      }
+      mousemove(px, py) {
+        if (isvolume) {
+          if (core.domStyle.isVertical) {
+            const time = Math.min(Math.max((px - 250) / 800, 0), 1);
+            audio.setVolume(time);
+            core.plugin.audioSystem.soundPlayer.setVolume(time);
+          } else {
+            const time = Math.min(Math.max((px - 100) / 600, 0), 1);
+            audio.setVolume(time);
+            core.plugin.audioSystem.soundPlayer.setVolume(time);
+          }
+        }
+      }
+
+      drawUI() {
+        //绘制页面
+        core.clearMap(music);
+        const bgVertical = core.material.images.images["bg_2010.webp"]; //竖屏背景
+        const bg = core.material.images.images["bg_5043.webp"]; //竖屏背景
+        if (core.domStyle.isVertical) {
+          //竖屏
+
+          core.fillRect(ctx, 0, 0, 1248, 2028, "#000000"); //黑色背景
+          ctx.globalAlpha = 0.3; //透明度
+          if (bgVertical)
+            ctx.drawImage(bgVertical, 0, 0, 1280, 1500, 0, 0, 1248, 2028); //绘制半透明背景图片
+          ctx.globalAlpha = 1; //恢复为不透明
+
+          core.setTextAlign(ctx, "center");
+          core.fillBoldText1(
+            ctx,
+            "◀离开",
+            110,
+            100,
+            "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+
+          ctx.strokeStyle = "#FFFFFF";
+          ctx.lineWidth = 3;
+          ctx.beginPath();
+          ctx.moveTo(100, 200);
+          ctx.lineTo(1148, 200);
+
+          ctx.stroke();
+          let posy = 300;
+          const indexList = this.musicMx[page];
+          core.setTextAlign(ctx, "left");
+          for (let i = 0; i < indexList.length; i++) {
+            const text = this.musicname[indexList[i]];
+            if (
+              this.musics.includes(this.musicMx[page][i]) ||
+              page !== this.musicMx.length - 1
+            ) {
+              core.fillBoldText1(
+                ctx,
+                text,
+                150,
+                posy - 30,
+                page === this.selection[0] && i === this.selection[1]
+                  ? "#FFFFFF"
+                  : "#444444",
+                "#000000",
+                6,
+                core.ui._buildFont(66, true)
+              );
+
+              ctx.strokeStyle = "#FFFFFF";
+              ctx.lineWidth = 3;
+              ctx.beginPath();
+              ctx.moveTo(100, posy);
+              ctx.lineTo(1148, posy);
+              ctx.stroke();
+            }
+            posy += 100;
+          }
+          ctx.beginPath();
+          ctx.moveTo(100, 1210);
+          ctx.lineTo(1148, 1210);
+          ctx.moveTo(100, 1200);
+          ctx.lineTo(1148, 1200);
+          ctx.stroke();
+
+          core.fillBoldText1(
+            ctx,
+            "上一页",
+            100,
+            1300,
+            page === 0 ? "#444444" : "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+
+          core.fillBoldText1(
+            ctx,
+            page + 1 + "/" + this.musicMx.length,
+            580,
+            1300,
+            "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+          core.fillBoldText1(
+            ctx,
+            "下一页",
+            950,
+            1300,
+            page === this.musicMx.length - 1 ? "#444444" : "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+
+          ctx.strokeStyle = "#ffffff";
+          ctx.lineWidth = 3;
+
+          core.fillBoldText(
+            ctx,
+            "|",
+            100,
+            1697,
+            "#FFFFFF",
+            6,
+            core.ui._buildFont(96, true)
+          );
+          core.fillBoldText(
+            ctx,
+            "◀",
+            115,
+            1700,
+            "#FFFFFF",
+            6,
+            core.ui._buildFont(96, true)
+          );
+
+          ctx.beginPath();
+          ctx.arc(505, 1670, 80, 0, 3 * Math.PI);
+          ctx.stroke();
+          core.fillText(
+            ctx,
+            "|",
+            835,
+            1697,
+            "#FFFFFF",
+            core.ui._buildFont(96, true)
+          );
+          core.fillText(
+            ctx,
+            "▶",
+            785,
+            1700,
+            "#FFFFFF",
+            core.ui._buildFont(96, true)
+          );
+
+          if (this.stop) {
+            core.fillText(
+              ctx,
+              "▶",
+              473,
+              1700,
+              "#FFFFFF",
+              core.ui._buildFont(96, true)
+            );
+          } else {
+            core.fillText(
+              ctx,
+              "||",
+              453,
+              1700,
+              "#FFFFFF",
+              core.ui._buildFont(96, true)
+            );
+          }
+
+          const img = core.material.images.images[this.type + ".webp"];
+          if (img) ctx.drawImage(img, 1000, 1555, 200, 200);
+          core.setTextAlign(ctx, "center");
+          ctx.font = "bold 52px Verdana";
+          ctx.fillText("当前歌曲", 625, 1397);
+          ctx.fillText(
+            this.musicname[this.musicMx[this.selection[0]][this.selection[1]]],
+            625,
+            1507
+          );
+
+          ctx.fillStyle = "#ffffff";
+          ctx.font = "bold 48px Verdana";
+          ctx.fillText("音量", 150, 1970);
+          ctx.lineWidth = 3;
+          ctx.beginPath();
+          ctx.moveTo(250, 1950);
+          ctx.lineTo(1050, 1950);
+          ctx.stroke();
+          ctx.strokeStyle = "#ffffff";
+          ctx.lineWidth = 9;
+          ctx.fillStyle = "rgba(255,255,255,0.5)";
+
+          ctx.beginPath();
+          ctx.moveTo(250, 1950);
+          ctx.lineTo(800 * audio.getVolume() + 250, 1950);
+          ctx.stroke();
+          ctx.beginPath();
+          ctx.arc(800 * audio.getVolume() + 250, 1950, 10, 0, 2 * Math.PI);
+          ctx.fill();
+          core.fillBoldText1(
+            ctx,
+            Math.floor(100 * audio.getVolume()),
+            1120,
+            1970,
+            "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(56, true)
+          );
+        } else {
+          //横屏
+
+          core.fillRect(ctx, 0, 0, 2028, 1248, "#000000"); //黑色背景
+          ctx.globalAlpha = 0.5; //透明度
+          if (bg) ctx.drawImage(bg, 0, 0, 1280, 720, 0, 0, 2028, 1248); //绘制半透明背景图片
+          ctx.globalAlpha = 1; //恢复为不透明
+          core.setTextAlign(ctx, "center");
+
+          core.fillBoldText1(
+            ctx,
+            "◀离开",
+            110,
+            100,
+            "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+
+          ctx.strokeStyle = "#FFFFFF";
+          ctx.lineWidth = 3;
+          ctx.beginPath();
+          ctx.moveTo(800, 100);
+          ctx.lineTo(800, 1148);
+          ctx.moveTo(900, 100);
+          ctx.lineTo(1900, 100);
+          ctx.stroke();
+          let posy = 200;
+          const indexList = this.musicMx[page];
+          core.setTextAlign(ctx, "left");
+          for (let i = 0; i < indexList.length; i++) {
+            const text = this.musicname[indexList[i]];
+            if (
+              this.musics.includes(this.musicMx[page][i]) ||
+              page !== this.musicMx.length - 1
+            ) {
+              core.fillBoldText1(
+                ctx,
+                text,
+                950,
+                posy - 30,
+                page === this.selection[0] && i === this.selection[1]
+                  ? "#FFFFFF"
+                  : "#444444",
+                "#000000",
+                6,
+                core.ui._buildFont(66, true)
+              );
+
+              ctx.strokeStyle = "#FFFFFF";
+              ctx.lineWidth = 3;
+              ctx.beginPath();
+              ctx.moveTo(900, posy);
+              ctx.lineTo(1900, posy);
+              ctx.stroke();
+            }
+            posy += 100;
+          }
+          core.fillBoldText1(
+            ctx,
+            "上一页",
+            1050,
+            1200 - 30,
+            page === 0 ? "#444444" : "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+
+          core.fillBoldText1(
+            ctx,
+            page + 1 + "/" + this.musicMx.length,
+            1350,
+            1200 - 30,
+            "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+          core.fillBoldText1(
+            ctx,
+            "下一页",
+            1550,
+            1200 - 30,
+            page === this.musicMx.length - 1 ? "#444444" : "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(66, true)
+          );
+          ctx.strokeStyle = "#ffffff";
+          ctx.lineWidth = 3;
+
+          core.fillText(
+            ctx,
+            "|",
+            55,
+            697,
+            "#FFFFFF",
+            core.ui._buildFont(96, true)
+          );
+          core.fillText(
+            ctx,
+            "◀",
+            70,
+            700,
+            "#FFFFFF",
+            core.ui._buildFont(96, true)
+          );
+
+          ctx.beginPath();
+          ctx.arc(295, 670, 80, 0, 2 * Math.PI);
+          ctx.stroke();
+          if (this.stop) {
+            core.fillText(
+              ctx,
+              "▶",
+              265,
+              700,
+              "#FFFFFF",
+              core.ui._buildFont(96, true)
+            );
+          } else {
+            core.fillText(
+              ctx,
+              "||",
+              245,
+              700,
+              "#FFFFFF",
+              core.ui._buildFont(96, true)
+            );
+          }
+          core.fillText(
+            ctx,
+            "|",
+            495,
+            697,
+            "#FFFFFF",
+            core.ui._buildFont(96, true)
+          );
+          core.fillText(
+            ctx,
+            "▶",
+            450,
+            700,
+            "#FFFFFF",
+            core.ui._buildFont(96, true)
+          );
+
+          ctx.font = "bold 48px Verdana";
+          ctx.fillText("音量", 350, 900);
+          ctx.beginPath();
+          ctx.moveTo(100, 1000);
+          ctx.lineTo(700, 1000);
+          ctx.stroke();
+          ctx.strokeStyle = "#ffffff";
+          ctx.lineWidth = 9;
+          ctx.fillStyle = "rgba(255,255,255,0.5)";
+
+          ctx.beginPath();
+          ctx.moveTo(100, 1000);
+          ctx.lineTo(600 * audio.getVolume() + 100, 1000);
+          ctx.stroke();
+          ctx.beginPath();
+          ctx.arc(600 * audio.getVolume() + 100, 1000, 10, 0, 2 * Math.PI);
+          ctx.fill();
+          core.fillBoldText1(
+            ctx,
+            Math.floor(100 * audio.getVolume()),
+            720,
+            1010,
+            "#FFFFFF",
+            "#000000",
+            6,
+            core.ui._buildFont(56, true)
+          );
+          const img = core.material.images.images[this.type + ".webp"];
+          if (img) ctx.drawImage(img, 580, 560, 200, 200);
+          core.setTextAlign(ctx, "center");
+          ctx.font = "bold 48px Verdana";
+          ctx.fillText("当前歌曲", 400, 297);
+          ctx.fillText(
+            this.musicname[this.musicMx[this.selection[0]][this.selection[1]]],
+            400,
+            397
+          );
+        }
+      }
+    }
+    core.ui.music = new musicclass();
+    main.dom.musicMode.onclick = function () {
+      //点击开始页面的CG MODE进入cg回廊
+      if (
+        (core.getLocalStorage("musics") &&
+          core.getLocalStorage("musics").length === 0) ||
+        !core.getLocalStorage("musics")
+      )
+        core.setLocalStorage("musics", ["theme.mp3"]);
+      core.ui.music.musics = core.getLocalStorage("musics");
+      core.playBgm(
+        main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+          main.core.ui.music.selection[1]
+        ],
+        0
+      );
+
+      const arr = main.core.ui.music.musicMx.flat(Infinity);
+      main.core.ui.music.randomList = shuffle(arr);
+      main.core.ui.music.random = main.core.ui.music.randomList.indexOf(
+        main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+          main.core.ui.music.selection[1]
+        ]
+      );
+      page = 0;
+      music.style.display = "block";
+      let time = 0;
+      core.registerAnimationFrame("music", null, (temptime) => {
+        if (temptime > time + 1000 / 60) {
+          time = temptime;
+          main.core.ui.music.update();
+          const duration =
+            core.plugin.audioSystem.bgmController.player.getRoute(
+              "bgms." +
+                main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                  main.core.ui.music.selection[1]
+                ]
+            ).duration;
+
+          const currentTime =
+            core.plugin.audioSystem.bgmController.player.getRoute(
+              "bgms." +
+                main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
+                  main.core.ui.music.selection[1]
+                ]
+            ).currentTime;
+          if (currentTime && duration && duration - currentTime < 0.05) {
+            if (core.domStyle.isVertical) {
+              core.ui.music.mousedown(830, 1770);
+            } else {
+              core.ui.music.mousedown(475, 765);
             }
           }
         }
-      );
+      });
+    };
+    loader.prototype.loadOneMusic = function (name) {
+      /* var music = new Audio();
+		music.preload = "none";
+		if (main.bgmRemote)
+		  music.src = main.bgmRemoteRoot + core.firstData.name + "/" + name;
+		else music.src = "project/bgms/" + name;
+		music.loop = "loop";
+		core.material.bgms[name] = music;*/
     };
   },
-    "musicMode": function () {
-	// 在此增加新插件
-	const music = document.createElement("canvas");
-	music.style.position = "absolute";
-	music.style.zIndex = 300;
-	music.style.display = "none";
-	music.id = "music";
-	main.dom.gameGroup.insertAdjacentElement("afterend", music);
-	music.style.top = "50%";
-	music.style.left = "50%";
-	music.style.transform = "translate(-50%,-50%)";
-	const ctx = music.getContext("2d");
-	main.dom.music = music;
-
-	const audio = core.plugin.audioSystem.bgmController;
-
-	let page = 0; //初始页面
-
-	let isvolume = false;
-
-	function shuffle(arr) {
-		let n = arr.length,
-			random;
-		while (n) {
-			random = (Math.random() * n--) >>> 0;
-			[arr[n], arr[random]] = [arr[random], arr[n]];
-		}
-		return arr;
-	}
-	music.addEventListener("mousedown", function (e) {
-		e.stopPropagation();
-		const left = core.dom.gameGroup.offsetLeft;
-		const top = core.dom.gameGroup.offsetTop;
-		const px = Math.floor((e.clientX - left) / core.domStyle.scale),
-			py = Math.floor((e.clientY - top) / core.domStyle.scale);
-		core.ui.music.mousedown(px * 3, py * 3);
-	});
-	music.addEventListener("mousemove", function (e) {
-		e.stopPropagation();
-		const left = core.dom.gameGroup.offsetLeft;
-		const top = core.dom.gameGroup.offsetTop;
-		const px = Math.floor((e.clientX - left) / core.domStyle.scale),
-			py = Math.floor((e.clientY - top) / core.domStyle.scale);
-		core.ui.music.mousemove(px * 3, py * 3);
-	});
-	music.addEventListener("mouseup", function (e) {
-		e.stopPropagation();
-
-		isvolume = false;
-	});
-	music.addEventListener("mouseleave", function (e) {
-		e.stopPropagation();
-
-		isvolume = false;
-	});
-	music.addEventListener("touchstart", function (e) {
-		e.preventDefault();
-		const left = core.dom.gameGroup.offsetLeft;
-		const top = core.dom.gameGroup.offsetTop;
-		const px = Math.floor(
-				(e.touches[0].clientX - left) / core.domStyle.scale
-			),
-			py = Math.floor((e.touches[0].clientY - top) / core.domStyle.scale);
-		core.ui.music.mousedown(px * 3, py * 3);
-	});
-	music.addEventListener("touchmove", function (e) {
-		e.stopPropagation();
-		const left = core.dom.gameGroup.offsetLeft;
-		const top = core.dom.gameGroup.offsetTop;
-		const px = Math.floor(
-				(e.touches[0].clientX - left) / core.domStyle.scale
-			),
-			py = Math.floor((e.touches[0].clientY - top) / core.domStyle.scale);
-		core.ui.music.mousemove(px * 3, py * 3);
-	});
-	music.addEventListener("touchend", function (e) {
-		e.stopPropagation();
-
-		isvolume = false;
-	});
-	music.addEventListener("touchcancel", function (e) {
-		e.stopPropagation();
-
-		isvolume = false;
-	});
-
-	class musicclass {
-		constructor() {
-			//music列表
-			//需全塔属性注册并保存在bgms文件夹,每个数组为显示的一页内容
-			this.musicMx = [
-				["Asphodelus_Ceui.opus", "Blind_Alley.opus"],
-				["Crawler.opus", "op.opus", "theme.opus"],
-			];
-			//音乐别名(将在播放器内显示的音乐名,music列表内的都要有对应歌名)
-			this.musicname = {
-				"Asphodelus_Ceui.opus": "Asphodelus",
-				"Blind_Alley.opus": "Blind",
-				"Crawler.opus": "Crawler",
-				"op.opus": "op",
-				"theme.opus": "theme",
-			};
-			this.selection = [0, 0];
-			this.stop = false;
-			this.type = "xunhuan";
-			this.randomList = [];
-			this.random = 0;
-		}
-
-		//更新
-		update() {
-			this.background();
-			this.drawUI();
-		}
-		background() {
-			//画布大小设置
-			if (core.domStyle.isVertical) {
-				music.width = 1248;
-				music.height = 2028;
-			} else {
-				music.width = 2028;
-				music.height = 1248;
-			}
-		}
-		mousedown(px, py) {
-			//鼠标按下时
-
-			const makeBox = ([x, y], [w, h]) => {
-				return [
-					[x, y],
-					[x + w, y + h],
-				];
-			};
-			const inRect = ([x, y], [
-				[sx, sy],
-				[dx, dy]
-			]) => {
-				return sx <= x && x <= dx && sy <= y && y <= dy;
-			};
-			const pos = [px, py];
-			const backbox = makeBox([15, 35], [210, 90]);
-			if (inRect(pos, backbox)) {
-				//离开按钮是一致的,其余的记区分横竖屏
-				music.style.display = "none";
-				core.clearMap(ctx);
-
-				core.unregisterAnimationFrame("music");
-				core.restart();
-
-				return;
-			}
-			if (core.domStyle.isVertical) {
-				//竖屏
-
-				const pageupbox = makeBox([100, 1230], [200, 100]);
-				const pagedownbox = makeBox([950, 1230], [200, 100]);
-				const musicbox = makeBox(
-					[100, 200],
-					[1048, this.musicMx[page].length * 100]
-				);
-				const beforebox = makeBox([120, 1620], [100, 100]);
-				const afterbox = makeBox([780, 1620], [100, 100]);
-				const playbox = makeBox([420, 1580], [200, 200]);
-				const typebox = makeBox([1040, 1600], [120, 120]);
-
-				const volumebox = makeBox([250, 1940], [1050, 20]);
-				if (inRect(pos, pageupbox)) {
-					if (page !== 0) page -= 1;
-					return;
-				}
-				if (inRect(pos, pagedownbox)) {
-					if (page !== this.musicMx.length - 1) page += 1;
-					return;
-				}
-				if (inRect(pos, playbox)) {
-					if (this.stop) {
-						this.stop = !this.stop;
-
-						core.resumeBgm();
-					} else {
-						this.stop = !this.stop;
-						core.pauseBgm();
-					}
-					return;
-				}
-				if (inRect(pos, beforebox)) {
-					this.stop = false;
-					switch (this.type) {
-					case "danqu":
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-
-						page = this.selection[0];
-
-						break;
-					case "xunhuan":
-						if (this.selection[1] === 0) {
-							if (this.selection[0] === 0) {
-								this.selection[0] = this.musicMx.length - 1;
-								this.selection[1] =
-									this.musicMx[this.selection[0]].length - 1;
-							} else {
-								this.selection[0] -= 1;
-								this.selection[1] =
-									this.musicMx[this.selection[0]].length - 1;
-							}
-						} else {
-							this.selection[1] -= 1;
-						}
-						this.randomList.indexOf(
-							this.musicMx[this.selection[0]][this.selection[1]]
-						);
-						page = this.selection[0];
-
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-
-						break;
-					case "suiji":
-						if (this.random > 0) {
-							this.random -= 1;
-						} else {
-							this.random = this.randomList.length - 1;
-						}
-						this.selection[0] = this.musicMx.findIndex((v) =>
-							v.includes(this.randomList[this.random])
-						);
-						this.selection[1] = this.musicMx[this.selection[0]].indexOf(
-							this.randomList[this.random]
-						);
-
-						page = this.selection[0];
-
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-
-						break;
-					}
-					return;
-				}
-				if (inRect(pos, afterbox)) {
-					this.stop = false;
-					switch (this.type) {
-					case "danqu":
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-
-						page = this.selection[0];
-						break;
-					case "xunhuan":
-						if (
-							this.selection[1] ===
-							this.musicMx[this.selection[0]].length - 1
-						) {
-							if (this.selection[0] === this.musicMx.length - 1) {
-								this.selection[0] = 0;
-								this.selection[1] = 0;
-							} else {
-								this.selection[0] += 1;
-								this.selection[1] = 0;
-							}
-						} else {
-							this.selection[1] += 1;
-						}
-						this.random = this.randomList.indexOf(
-							this.musicMx[this.selection[0]][this.selection[1]]
-						);
-						page = this.selection[0];
-
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-
-						break;
-					case "suiji":
-						if (this.random < this.randomList.length - 1) {
-							this.random += 1;
-						} else {
-							this.random = 0;
-						}
-						this.selection[0] = this.musicMx.findIndex((v) =>
-							v.includes(this.randomList[this.random])
-						);
-						this.selection[1] = this.musicMx[this.selection[0]].indexOf(
-							this.randomList[this.random]
-						);
-
-						page = this.selection[0];
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-
-						break;
-					}
-					return;
-				}
-				if (inRect(pos, typebox)) {
-					switch (this.type) {
-					case "danqu":
-						this.type = "xunhuan";
-						break;
-					case "xunhuan":
-						this.type = "suiji";
-						break;
-					case "suiji":
-						this.type = "danqu";
-						break;
-					}
-					return;
-				}
-				if (inRect(pos, musicbox)) {
-					const index = Math.floor((py - 200) / 100);
-					if (page !== this.selection[0] || index !== this.selection[1]) {
-						this.selection[0] = page;
-						this.selection[1] = index;
-						this.randomList.indexOf(
-							this.musicMx[this.selection[0]][this.selection[1]]
-						);
-
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-
-						this.stop = false;
-					} else {
-						if (this.stop) {
-							this.stop = !this.stop;
-							core.resumeBgm();
-						} else {
-							this.stop = !this.stop;
-							core.pauseBgm();
-						}
-					}
-					return;
-				}
-
-				if (inRect(pos, volumebox)) {
-					const time = Math.min(Math.max((px - 250) / 800, 0), 1);
-					audio.setVolume(time);
-					core.plugin.audioSystem.soundPlayer.setVolume(time);
-					isvolume = true;
-				}
-			} else {
-				//横屏
-				const pageupbox = makeBox([1050, 1100], [200, 100]);
-				const pagedownbox = makeBox([1550, 1100], [200, 100]);
-				const musicbox = makeBox(
-					[900, 100],
-					[1000, this.musicMx[page].length * 100]
-				);
-				const beforebox = makeBox([60, 620], [100, 100]);
-				const afterbox = makeBox([450, 620], [100, 100]);
-				const playbox = makeBox([200, 570], [200, 200]);
-				const typebox = makeBox([620, 600], [120, 120]);
-
-				const volumebox = makeBox([100, 990], [600, 20]);
-				if (inRect(pos, pageupbox)) {
-					if (page !== 0) page -= 1;
-					return;
-				}
-				if (inRect(pos, pagedownbox)) {
-					if (page !== this.musicMx.length - 1) page += 1;
-					return;
-				}
-				if (inRect(pos, playbox)) {
-					if (this.stop) {
-						this.stop = !this.stop;
-						core.resumeBgm();
-					} else {
-						this.stop = !this.stop;
-						core.pauseBgm();
-					}
-					return;
-				}
-				if (inRect(pos, beforebox)) {
-					this.stop = false;
-					switch (this.type) {
-					case "danqu":
-						if (!this.stop)
-							core.playBgm(
-								main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-									main.core.ui.music.selection[1]
-								]
-							);
-
-						page = this.selection[0];
-
-						break;
-					case "xunhuan":
-						if (this.selection[1] === 0) {
-							if (this.selection[0] === 0) {
-								this.selection[0] = this.musicMx.length - 1;
-								this.selection[1] =
-									this.musicMx[this.selection[0]].length - 1;
-							} else {
-								this.selection[0] -= 1;
-								this.selection[1] =
-									this.musicMx[this.selection[0]].length - 1;
-							}
-						} else {
-							this.selection[1] -= 1;
-						}
-						this.random = this.randomList.indexOf(
-							this.musicMx[this.selection[0]][this.selection[1]]
-						);
-						page = this.selection[0];
-
-						if (!this.stop)
-							core.playBgm(
-								main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-									main.core.ui.music.selection[1]
-								]
-							);
-
-						break;
-					case "suiji":
-						if (this.random > 0) {
-							this.random -= 1;
-						} else {
-							this.random = this.randomList.length - 1;
-						}
-						this.selection[0] = this.musicMx.findIndex((v) =>
-							v.includes(this.randomList[this.random])
-						);
-						this.selection[1] = this.musicMx[this.selection[0]].indexOf(
-							this.randomList[this.random]
-						);
-
-						page = this.selection[0];
-
-						if (!this.stop)
-							core.playBgm(
-								main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-									main.core.ui.music.selection[1]
-								]
-							);
-
-						break;
-					}
-					return;
-				}
-				if (inRect(pos, afterbox)) {
-					this.stop = false;
-					switch (this.type) {
-					case "danqu":
-						if (!this.stop)
-							core.playBgm(
-								main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-									main.core.ui.music.selection[1]
-								]
-							);
-
-						page = this.selection[0];
-						break;
-					case "xunhuan":
-						if (
-							this.selection[1] ===
-							this.musicMx[this.selection[0]].length - 1
-						) {
-							if (this.selection[0] === this.musicMx.length - 1) {
-								this.selection[0] = 0;
-								this.selection[1] = 0;
-							} else {
-								this.selection[0] += 1;
-								this.selection[1] = 0;
-							}
-						} else {
-							this.selection[1] += 1;
-						}
-						this.randomList.findIndex(
-							(v) =>
-							v === this.musicMx[this.selection[0]][this.selection[1]]
-						);
-						page = this.selection[0];
-
-						if (!this.stop)
-							core.playBgm(
-								main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-									main.core.ui.music.selection[1]
-								]
-							);
-
-						break;
-					case "suiji":
-						if (this.random < this.randomList.length - 1) {
-							this.random += 1;
-						} else {
-							this.random = 0;
-						}
-						this.selection[0] = this.musicMx.findIndex((v) =>
-							v.includes(this.randomList[this.random])
-						);
-						this.selection[1] = this.musicMx[this.selection[0]].indexOf(
-							main.core.ui.music.randomList[main.core.ui.music.random]
-						);
-
-						page = this.selection[0];
-						if (!this.stop)
-							core.playBgm(
-								main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-									main.core.ui.music.selection[1]
-								]
-							);
-
-						break;
-					}
-					return;
-				}
-				if (inRect(pos, typebox)) {
-					switch (this.type) {
-					case "danqu":
-						this.type = "xunhuan";
-						break;
-					case "xunhuan":
-						this.type = "suiji";
-						break;
-					case "suiji":
-						this.type = "danqu";
-						break;
-					}
-					return;
-				}
-				if (inRect(pos, musicbox)) {
-					const index = Math.floor((py - 100) / 100);
-					if (page !== this.selection[0] || index !== this.selection[1]) {
-						this.selection[0] = page;
-						this.selection[1] = index;
-						this.randomList.indexOf(
-							(v) => v === this.musicMx[this.selection[0]][this.selection[1]]
-						);
-
-						core.playBgm(
-							main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-								main.core.ui.music.selection[1]
-							]
-						);
-						this.stop = false;
-					} else {
-						if (this.stop) {
-							this.stop = !this.stop;
-							core.resumeBgm();
-						} else {
-							this.stop = !this.stop;
-							core.pauseBgm();
-						}
-					}
-					return;
-				}
-
-				if (inRect(pos, volumebox)) {
-					const time = Math.min(Math.max((px - 100) / 600, 0), 1);
-					audio.setVolume(time);
-					core.plugin.audioSystem.soundPlayer.setVolume(time);
-					isvolume = true;
-				}
-			}
-		}
-		mousemove(px, py) {
-			if (isvolume) {
-				if (core.domStyle.isVertical) {
-					const time = Math.min(Math.max((px - 250) / 800, 0), 1);
-					audio.setVolume(time);
-					core.plugin.audioSystem.soundPlayer.setVolume(time);
-				} else {
-					const time = Math.min(Math.max((px - 100) / 600, 0), 1);
-					audio.setVolume(time);
-					core.plugin.audioSystem.soundPlayer.setVolume(time);
-				}
-			}
-		}
-
-		drawUI() {
-			//绘制页面
-			core.clearMap(music);
-			const bgVertical = core.material.images.images["bg_2010.webp"]; //竖屏背景
-			const bg = core.material.images.images["bg_5043.webp"]; //竖屏背景
-			if (core.domStyle.isVertical) {
-				//竖屏
-
-				core.fillRect(ctx, 0, 0, 1248, 2028, "#000000"); //黑色背景
-				ctx.globalAlpha = 0.3; //透明度
-				if (bgVertical)
-					ctx.drawImage(bgVertical, 0, 0, 1280, 1500, 0, 0, 1248, 2028); //绘制半透明背景图片
-				ctx.globalAlpha = 1; //恢复为不透明
-
-				core.setTextAlign(ctx, "center");
-				core.fillBoldText1(
-					ctx,
-					"◀离开",
-					110,
-					100,
-					"#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-
-				ctx.strokeStyle = "#FFFFFF";
-				ctx.lineWidth = 3;
-				ctx.beginPath();
-				ctx.moveTo(100, 200);
-				ctx.lineTo(1148, 200);
-
-				ctx.stroke();
-				let posy = 300;
-				const indexList = this.musicMx[page];
-				core.setTextAlign(ctx, "left");
-				for (let i = 0; i < indexList.length; i++) {
-					const text = this.musicname[indexList[i]];
-					core.fillBoldText1(
-						ctx,
-						text,
-						150,
-						posy - 30,
-						page === this.selection[0] && i === this.selection[1] ?
-						"#FFFFFF" :
-						"#444444",
-						"#000000",
-						6,
-						core.ui._buildFont(66, true)
-					);
-					ctx.strokeStyle = "#FFFFFF";
-					ctx.lineWidth = 3;
-					ctx.beginPath();
-					ctx.moveTo(100, posy);
-					ctx.lineTo(1148, posy);
-					ctx.stroke();
-					posy += 100;
-				}
-				ctx.beginPath();
-				ctx.moveTo(100, 1210);
-				ctx.lineTo(1148, 1210);
-				ctx.moveTo(100, 1200);
-				ctx.lineTo(1148, 1200);
-				ctx.stroke();
-
-				core.fillBoldText1(
-					ctx,
-					"上一页",
-					100,
-					1300,
-					page === 0 ? "#444444" : "#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-
-				core.fillBoldText1(
-					ctx,
-					page + 1 + "/" + this.musicMx.length,
-					580,
-					1300,
-					"#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-				core.fillBoldText1(
-					ctx,
-					"下一页",
-					950,
-					1300,
-					page === this.musicMx.length - 1 ? "#444444" : "#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-
-				ctx.strokeStyle = "#ffffff";
-				ctx.lineWidth = 3;
-
-				ctx.fillStyle = "#ffffff";
-				ctx.font = "bold 96px Verdana";
-				ctx.fillText("|", 100, 1697);
-				ctx.fillText("◀", 115, 1700);
-				ctx.beginPath();
-				ctx.arc(505, 1670, 80, 0, 3 * Math.PI);
-				ctx.stroke();
-				ctx.fillText("|", 835, 1697);
-				ctx.fillText("▶", 785, 1700);
-				if (this.stop) {
-					ctx.fillText("▶", 473, 1697);
-				} else {
-					ctx.fillText("||", 453, 1694);
-				}
-
-				const img = core.material.images.images[this.type + ".webp"];
-				if (img) ctx.drawImage(img, 1000, 1555, 200, 200);
-				core.setTextAlign(ctx, "center");
-				ctx.font = "bold 52px Verdana";
-				ctx.fillText("当前歌曲", 625, 1397);
-				ctx.fillText(
-					this.musicname[this.musicMx[this.selection[0]][this.selection[1]]],
-					625,
-					1507
-				);
-
-				ctx.fillStyle = "#ffffff";
-				ctx.font = "bold 48px Verdana";
-				ctx.fillText("音量", 150, 1970);
-				ctx.lineWidth = 3;
-				ctx.beginPath();
-				ctx.moveTo(250, 1950);
-				ctx.lineTo(1050, 1950);
-				ctx.stroke();
-				ctx.strokeStyle = "#ffffff";
-				ctx.lineWidth = 9;
-				ctx.fillStyle = "rgba(255,255,255,0.5)";
-
-				ctx.beginPath();
-				ctx.moveTo(250, 1950);
-				ctx.lineTo(800 * audio.getVolume() + 250, 1950);
-				ctx.stroke();
-				ctx.beginPath();
-				ctx.arc(800 * audio.getVolume() + 250, 1950, 10, 0, 2 * Math.PI);
-				ctx.fill();
-				core.fillBoldText1(
-					ctx,
-					Math.floor(100 * audio.getVolume()),
-					1120,
-					1970,
-					"#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(56, true)
-				);
-			} else {
-				//横屏
-
-				core.fillRect(ctx, 0, 0, 2028, 1248, "#000000"); //黑色背景
-				ctx.globalAlpha = 0.5; //透明度
-				if (bg) ctx.drawImage(bg, 0, 0, 1280, 720, 0, 0, 2028, 1248); //绘制半透明背景图片
-				ctx.globalAlpha = 1; //恢复为不透明
-				core.setTextAlign(ctx, "center");
-
-				core.fillBoldText1(
-					ctx,
-					"◀离开",
-					110,
-					100,
-					"#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-
-				ctx.strokeStyle = "#FFFFFF";
-				ctx.lineWidth = 3;
-				ctx.beginPath();
-				ctx.moveTo(800, 100);
-				ctx.lineTo(800, 1148);
-				ctx.moveTo(900, 100);
-				ctx.lineTo(1900, 100);
-				ctx.stroke();
-				let posy = 200;
-				const indexList = this.musicMx[page];
-				core.setTextAlign(ctx, "left");
-				for (let i = 0; i < indexList.length; i++) {
-					const text = this.musicname[indexList[i]];
-					core.fillBoldText1(
-						ctx,
-						text,
-						950,
-						posy - 30,
-						page === this.selection[0] && i === this.selection[1] ?
-						"#FFFFFF" :
-						"#444444",
-						"#000000",
-						6,
-						core.ui._buildFont(66, true)
-					);
-					ctx.strokeStyle = "#FFFFFF";
-					ctx.lineWidth = 3;
-					ctx.beginPath();
-					ctx.moveTo(900, posy);
-					ctx.lineTo(1900, posy);
-					ctx.stroke();
-					posy += 100;
-				}
-				core.fillBoldText1(
-					ctx,
-					"上一页",
-					1050,
-					1200 - 30,
-					page === 0 ? "#444444" : "#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-
-				core.fillBoldText1(
-					ctx,
-					page + 1 + "/" + this.musicMx.length,
-					1350,
-					1200 - 30,
-					"#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-				core.fillBoldText1(
-					ctx,
-					"下一页",
-					1550,
-					1200 - 30,
-					page === this.musicMx.length - 1 ? "#444444" : "#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(66, true)
-				);
-				ctx.strokeStyle = "#ffffff";
-				ctx.lineWidth = 3;
-
-				ctx.fillStyle = "#ffffff";
-				ctx.font = "bold 96px Verdana";
-				ctx.fillText("|", 60, 697);
-				ctx.fillText("◀", 70, 700);
-				ctx.beginPath();
-				ctx.arc(295, 670, 80, 0, 2 * Math.PI);
-				ctx.stroke();
-				if (this.stop) {
-					ctx.fillText("▶", 245, 697);
-				} else {
-					ctx.fillText("||", 245, 694);
-				}
-
-				ctx.fillText("|", 490, 697);
-				ctx.fillText("▶", 450, 700);
-				ctx.font = "bold 48px Verdana";
-				ctx.fillText("音量", 350, 900);
-				ctx.beginPath();
-				ctx.moveTo(100, 1000);
-				ctx.lineTo(700, 1000);
-				ctx.stroke();
-				ctx.strokeStyle = "#ffffff";
-				ctx.lineWidth = 9;
-				ctx.fillStyle = "rgba(255,255,255,0.5)";
-
-				ctx.beginPath();
-				ctx.moveTo(100, 1000);
-				ctx.lineTo(600 * audio.getVolume() + 100, 1000);
-				ctx.stroke();
-				ctx.beginPath();
-				ctx.arc(600 * audio.getVolume() + 100, 1000, 10, 0, 2 * Math.PI);
-				ctx.fill();
-				core.fillBoldText1(
-					ctx,
-					Math.floor(100 * audio.getVolume()),
-					720,
-					1010,
-					"#FFFFFF",
-					"#000000",
-					6,
-					core.ui._buildFont(56, true)
-				);
-				const img = core.material.images.images[this.type + ".webp"];
-				if (img) ctx.drawImage(img, 580, 560, 200, 200);
-				core.setTextAlign(ctx, "center");
-				ctx.font = "bold 48px Verdana";
-				ctx.fillText("当前歌曲", 400, 297);
-				ctx.fillText(
-					this.musicname[this.musicMx[this.selection[0]][this.selection[1]]],
-					400,
-					397
-				);
-			}
-		}
-	}
-	core.ui.music = new musicclass();
-	main.dom.musicMode.onclick = function () {
-		//点击开始页面的CG MODE进入cg回廊
-
-		core.playBgm(
-			main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-				main.core.ui.music.selection[1]
-			]
-		);
-
-		const arr = main.core.ui.music.musicMx.flat(Infinity);
-		main.core.ui.music.randomList = shuffle(arr);
-		main.core.ui.music.random = main.core.ui.music.randomList.indexOf(
-			main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-				main.core.ui.music.selection[1]
-			]
-		);
-		page = 0;
-		music.style.display = "block";
-		let time = 0;
-		core.registerAnimationFrame("music", null, (temptime) => {
-			if (temptime > time + 1000 / 60) {
-				time = temptime;
-				main.core.ui.music.update();
-				const duration =
-					core.plugin.audioSystem.bgmController.player.getRoute(
-						"bgms." +
-						main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-							main.core.ui.music.selection[1]
-						]
-					).duration;
-
-				const currentTime =
-					core.plugin.audioSystem.bgmController.player.getRoute(
-						"bgms." +
-						main.core.ui.music.musicMx[main.core.ui.music.selection[0]][
-							main.core.ui.music.selection[1]
-						]
-					).currentTime;
-				if (currentTime && duration && duration - currentTime < 0.05) {
-					if (core.domStyle.isVertical) {
-						core.ui.music.mousedown(830, 1770);
-					} else {
-						core.ui.music.mousedown(475, 765);
-					}
-				}
-			}
-		});
-	};
-},
     "横屏切换": function () {
     // 在此增加新插件
     this.triggerFullscreen = async function (full) {
@@ -17369,200 +21019,5219 @@ var plugins_bb40132b_638b_4a9f_b028_d3fe47acc8d1 =
       });
     };
   },
-    "帧动画特效(游戏界面)": function () {
+    "intro&loop": function () {
     // 在此增加新插件
-    const animate2 = document.createElement("canvas"); //画布设置
-    animate2.style.zIndex = 71;
-    animate2.id = "animate2";
-    animate2.classList.add("gameCanvas", "anti-aliasing");
-    animate2.style.display = "block";
-    animate2.width = 416;
-    animate2.height = 416;
-    animate2.style.width = core.__PIXELS__ * core.domStyle.scale + "px";
-    animate2.style.height = core.__PIXELS__ * core.domStyle.scale + "px";
-    main.dom.animate2 = animate2;
-    const anctx = animate2.getContext("2d");
+    this.introAndLoop = function (intro, time, loop) {
+      core.playBgm(intro);
+      setTimeout(() => {
+        core.playBgm(loop);
+      }, time * 1000);
+    };
+  },
+    "开局选项悬停": function () {
+	// 在此增加新插件
+	main.dom.playGame.addEventListener("mouseenter", () => {
+		core.dom.playGame.style.backgroundColor = "#808080";
+	});
+	main.dom.playGame.addEventListener("mouseleave", () => {
+		core.dom.playGame.style.backgroundColor = "transparent";
+	});
+	main.dom.playGame.addEventListener("touchmove", () => {
+		core.dom.playGame.style.backgroundColor = "#808080";
+	});
+	main.dom.playGame.addEventListener("touchend", () => {
+		core.dom.playGame.style.backgroundColor = "transparent";
+	});
+	main.dom.playGame.addEventListener("touchcancel", () => {
+		core.dom.playGame.style.backgroundColor = "transparent";
+	});
 
-    main.dom.gameDraw.appendChild(animate2);
+	main.dom.loadGame.addEventListener("mouseenter", () => {
+		core.dom.loadGame.style.backgroundColor = "#808080";
+	});
+	main.dom.loadGame.addEventListener("mouseleave", () => {
+		core.dom.loadGame.style.backgroundColor = "transparent";
+	});
+	main.dom.loadGame.addEventListener("touchmove", () => {
+		core.dom.loadGame.style.backgroundColor = "#808080";
+	});
+	main.dom.loadGame.addEventListener("touchend", () => {
+		core.dom.loadGame.style.backgroundColor = "transparent";
+	});
+	main.dom.loadGame.addEventListener("touchcancel", () => {
+		core.dom.loadGame.style.backgroundColor = "transparent";
+	});
 
-    core.plugin.playing = new Set();
+	main.dom.CGMode.addEventListener("mouseenter", () => {
+		core.dom.CGMode.style.backgroundColor = "#808080";
+	});
+	main.dom.CGMode.addEventListener("mouseleave", () => {
+		core.dom.CGMode.style.backgroundColor = "transparent";
+	});
+	main.dom.CGMode.addEventListener("touchmove", () => {
+		core.dom.CGMode.style.backgroundColor = "#808080";
+	});
+	main.dom.CGMode.addEventListener("touchend", () => {
+		core.dom.CGMode.style.backgroundColor = "transparent";
+	});
+	main.dom.CGMode.addEventListener("touchcancel", () => {
+		core.dom.CGMode.style.backgroundColor = "transparent";
+	});
 
-    this.setanimate = function (
-      name,
-      px,
-      py,
-      width,
-      height,
-      allFarme,
-      imageList,
-      soundList
+	main.dom.musicMode.addEventListener("mouseenter", () => {
+		core.dom.musicMode.style.backgroundColor = "#808080";
+	});
+	main.dom.musicMode.addEventListener("mouseleave", () => {
+		core.dom.musicMode.style.backgroundColor = "transparent";
+	});
+	main.dom.musicMode.addEventListener("touchmove", () => {
+		core.dom.musicMode.style.backgroundColor = "#808080";
+	});
+	main.dom.musicMode.addEventListener("touchend", () => {
+		core.dom.musicMode.style.backgroundColor = "transparent";
+	});
+	main.dom.musicMode.addEventListener("touchcancel", () => {
+		core.dom.musicMode.style.backgroundColor = "transparent";
+	});
+
+	main.dom.replayGame.addEventListener("mouseenter", () => {
+		core.dom.replayGame.style.backgroundColor = "#808080";
+	});
+	main.dom.replayGame.addEventListener("mouseleave", () => {
+		core.dom.replayGame.style.backgroundColor = "transparent";
+	});
+	main.dom.replayGame.addEventListener("touchmove", () => {
+		core.dom.replayGame.style.backgroundColor = "#808080";
+	});
+	main.dom.replayGame.addEventListener("touchend", () => {
+		core.dom.replayGame.style.backgroundColor = "transparent";
+	});
+	main.dom.replayGame.addEventListener("touchcancel", () => {
+		core.dom.replayGame.style.backgroundColor = "transparent";
+	});
+},
+    "天气叠加": function () {
+	//使用方法:使用core.setWeather(天气,等级)来增加天气,使用core.setWeather()来清空天气
+	// 天气叠加功能
+
+	////// 更改天气效果 //////
+	control.prototype.setWeather = function (type, level) {
+		// 非雨雪
+		if (type == null) {
+			Object.keys(core.control.weathers).forEach(function (one) {
+				core.deleteCanvas("weather" + one);
+			});
+			core.animateFrame.weather.type = [];
+			core.animateFrame.weather.nodes = {};
+			core.animateFrame.weather.level = {};
+			core.animateFrame.weather.time = {};
+			return;
+		}
+		if (!core.animateFrame.weather.level || level == null)
+			core.animateFrame.weather.level = {};
+		if (!core.animateFrame.weather.type) core.animateFrame.weather.type = [];
+		level = core.clamp(parseInt(level) || 5, 1, 10);
+		// 当前天气:则忽略
+		if (
+			core.animateFrame.weather.type.includes(type) &&
+			level == core.animateFrame.weather.level[type]
+		)
+			return;
+		if (core.animateFrame.weather.nodes[type]) return;
+		// 计算当前的宽高
+		core.createCanvas(
+			"weather" + type,
+			0,
+			0,
+			core.__PIXELS__,
+			core.__PIXELS__,
+			80
+		);
+		core.animateFrame.weather.type.push(type);
+		core.animateFrame.weather.level[type] = level;
+		this._setWeather_createNodes(type, level);
+	};
+	control.prototype._setWeather_createNodes = function (type, level) {
+		var number =
+			level *
+			parseInt(
+				(20 * core.bigmap.width * core.bigmap.height) /
+				(core.__SIZE__ * core.__SIZE__)
+			);
+		if (!core.animateFrame.weather.nodes[type])
+			core.animateFrame.weather.nodes[type] = [];
+		switch (type) {
+		case "rain":
+			for (var a = 0; a < number; a++) {
+				core.animateFrame.weather.nodes.rain.push({
+					x: Math.random() * core.bigmap.width * 32,
+					y: Math.random() * core.bigmap.height * 32,
+					l: Math.random() * 2.5,
+					xs: -4 + Math.random() * 4 + 2,
+					ys: Math.random() * 10 + 10,
+				});
+			}
+			break;
+		case "snow":
+			for (var a = 0; a < number; a++) {
+				core.animateFrame.weather.nodes.snow.push({
+					x: Math.random() * core.bigmap.width * 32,
+					y: Math.random() * core.bigmap.height * 32,
+					r: Math.random() * 5 + 1,
+					d: Math.random() * Math.min(level, 200),
+				});
+			}
+			break;
+		case "fog":
+			if (core.animateFrame.weather.fog) {
+				core.animateFrame.weather.nodes[type] = [{
+					level: number,
+					x: 0,
+					y: -core.__PIXELS__ / 2,
+					dx: -Math.random() * 1.5,
+					dy: Math.random(),
+					delta: 0.001,
+				}, ];
+			}
+			break;
+		case "cloud":
+			if (core.animateFrame.weather.cloud) {
+				core.animateFrame.weather.nodes[type] = [{
+					level: number,
+					x: 0,
+					y: -core.__PIXELS__ / 2,
+					dx: -Math.random() * 1.5,
+					dy: Math.random(),
+					delta: 0.001,
+				}, ];
+			}
+			break;
+		case "sun":
+			if (core.animateFrame.weather.sun) {
+				// 直接绘制
+				core.clearMap("weather" + type);
+
+				core.drawImage(
+					"weather" + type,
+					core.animateFrame.weather.sun,
+					0,
+					0,
+					core.animateFrame.weather.sun.width,
+					core.animateFrame.weather.sun.height,
+					0,
+					0,
+					core.__PIXELS__,
+					core.__PIXELS__
+				);
+				core.setAlpha("weather" + type, level / 10);
+				core.animateFrame.weather.nodes.sun = [{ level: level, opacity: level / 10, delta: 0.01 }]
+			}
+			break;
+		}
+	};
+	core.registerAnimationFrame("weather", true, function (timestamp) {
+		var weather = core.animateFrame.weather;
+		if (!weather.type) return;
+		weather.type.forEach(function (one) {
+			if (
+				timestamp - weather.time[one] <= 30 ||
+				!core.dymCanvas["weather" + one]
+			)
+				return;
+			core.control["_animationFrame_weather_" + one]();
+			weather.time[one] = timestamp;
+		});
+	});
+	// 晴
+	control.prototype._animationFrame_weather_sun = function () {
+
+		var node = core.animateFrame.weather.nodes.sun[0]
+		var opacity = node.opacity + node.delta;
+
+		if (opacity > node.level / 11 + 0.3 || opacity < node.level / 11 - 0.3)
+			node.delta = -node.delta;
+		node.opacity = opacity;
+		core.setOpacity("weathersun", core.clamp(opacity, 0, 1));
+
+	};
+	// 雨
+	control.prototype._animationFrame_weather_rain = function () {
+		var ctx = core.dymCanvas.weatherrain,
+			ox = core.bigmap.offsetX,
+			oy = core.bigmap.offsetY;
+		core.clearMap("weatherrain");
+		ctx.strokeStyle = "rgba(174,194,224,0.8)";
+		ctx.lineWidth = 1;
+		ctx.lineCap = "round";
+		core.animateFrame.weather.nodes.rain.forEach(function (p) {
+			ctx.beginPath();
+			ctx.moveTo(p.x - ox, p.y - oy);
+			ctx.lineTo(p.x + p.l * p.xs - ox, p.y + p.l * p.ys - oy);
+			ctx.stroke();
+			p.x += p.xs;
+			p.y += p.ys;
+			if (p.x > core.bigmap.width * 32 || p.y > core.bigmap.height * 32) {
+				p.x = Math.random() * core.bigmap.width * 32;
+				p.y = -10;
+			}
+		});
+		ctx.fill();
+	};
+	// 雪
+	control.prototype._animationFrame_weather_snow = function () {
+		var ctx = core.dymCanvas.weathersnow,
+			ox = core.bigmap.offsetX,
+			oy = core.bigmap.offsetY;
+		core.clearMap("weathersnow");
+		ctx.fillStyle = "rgba(255, 255, 255, 0.8)";
+		ctx.beginPath();
+		if (!core.animateFrame.weather.data) core.animateFrame.weather.data = {};
+		core.animateFrame.weather.data.snow =
+			core.animateFrame.weather.data.snow || 0;
+		core.animateFrame.weather.data.snow += 0.01;
+		var angle = core.animateFrame.weather.data.snow;
+		core.animateFrame.weather.nodes.snow.forEach(function (p) {
+			ctx.moveTo(p.x - ox, p.y - oy);
+			ctx.arc(p.x - ox, p.y - oy, p.r, 0, Math.PI * 2, true);
+			// update
+			p.x += Math.sin(angle) * core.animateFrame.weather.level.snow;
+			p.y += Math.cos(angle + p.d) + 1 + p.r / 2;
+			if (
+				p.x > core.bigmap.width * 32 + 5 ||
+				p.x < -5 ||
+				p.y > core.bigmap.height * 32
+			) {
+				if (Math.random() > 1 / 3) {
+					p.x = Math.random() * core.bigmap.width * 32;
+					p.y = -10;
+				} else {
+					if (Math.sin(angle) > 0) p.x = -5;
+					else p.x = core.bigmap.width * 32 + 5;
+					p.y = Math.random() * core.bigmap.height * 32;
+				}
+			}
+		});
+		ctx.fill();
+	};
+	// 图片天气
+	control.prototype.__animateFrame_weather_image = function (image, type) {
+		if (!image) return;
+		var node = core.animateFrame.weather.nodes[type][0];
+		core.setAlpha("weather" + type, node.level / 500);
+		var wind = 1.5;
+		var width = image.width,
+			height = image.height;
+		node.x += node.dx * wind;
+		node.y += (2 * node.dy - 1) * wind;
+		if (node.x + 3 * width <= core.__PIXELS__) {
+			node.x += 4 * width;
+			while (node.x > 0) node.x -= width;
+		}
+		node.dy += node.delta;
+		if (node.dy >= 1) {
+			node.delta = -0.001;
+		} else if (node.dy <= 0) {
+			node.delta = 0.001;
+		}
+		if (node.y + 3 * height <= core.__PIXELS__) {
+			node.y += 4 * height;
+			while (node.y > 0) node.y -= height;
+		} else if (node.y >= 0) {
+			node.y -= height;
+		}
+		for (var i = 0; i < 3; ++i) {
+			for (var j = 0; j < 3; ++j) {
+				if (
+					node.x + (i + 1) * width <= 0 ||
+					node.x + i * width >= core.__PIXELS__ ||
+					node.y + (j + 1) * height <= 0 ||
+					node.y + j * height >= core.__PIXELS__
+				)
+					continue;
+				core.drawImage(
+					"weather" + type,
+					image,
+					node.x + i * width,
+					node.y + j * height
+				);
+			}
+		}
+		core.setAlpha("weather" + type, 1);
+	};
+	// 雾
+	control.prototype._animationFrame_weather_fog = function () {
+		core.clearMap("weatherfog");
+		this.__animateFrame_weather_image(core.animateFrame.weather.fog, "fog");
+	};
+	// 云
+	control.prototype._animationFrame_weather_cloud = function () {
+		core.clearMap("weathercloud");
+		this.__animateFrame_weather_image(
+			core.animateFrame.weather.cloud,
+			"cloud"
+		);
+	}
+
+},
+    "回合战斗动画": function () {
+	// 在此增加新插件
+	const animateAttack = document.createElement("canvas"); //画布设置
+	animateAttack.style.zIndex = 80;
+	animateAttack.id = "animateAttack";
+	animateAttack.classList.add("gameCanvas", "anti-aliasing");
+	animateAttack.style.display = "block";
+	animateAttack.width = 416;
+	animateAttack.height = 416;
+	animateAttack.style.width = core.__PIXELS__ * core.domStyle.scale + "px";
+	animateAttack.style.height = core.__PIXELS__ * core.domStyle.scale + "px";
+	main.dom.animateAttack = animateAttack;
+	const ctx = animateAttack.getContext("2d");
+	const changeY = -30
+	let easy = false;
+	const { imagelighter } = core.plugin.utils;
+	main.dom.gameDraw.appendChild(animateAttack);
+	const { lcm, gcd } = core.plugin.utils;
+	const animate2 = document.createElement("canvas"); //画布设置
+	animate2.style.zIndex = 91;
+	animate2.id = "animate2";
+	animate2.classList.add("gameCanvas", "anti-aliasing");
+	animate2.style.display = "block";
+	animate2.width = 416;
+	animate2.height = 416;
+	animate2.style.width = core.__PIXELS__ * core.domStyle.scale + "px";
+	animate2.style.height = core.__PIXELS__ * core.domStyle.scale + "px";
+	main.dom.animate2 = animate2;
+	const anctx = animate2.getContext("2d");
+
+	main.dom.gameDraw.appendChild(animate2);
+	let a = []
+
+	function drawAnimate(name, centerX, centerY) {
+		name = core.getMappedName(name);
+
+		// 正在播放录像:不显示动画
+		if (core.isReplaying() || !core.material.animates[name] || centerX == null || centerY == null) {
+			return -1;
+		}
+
+		// 开始绘制
+		var animate = core.material.animates[name];
+		animate.se = animate.se || {};
+		if (typeof animate.se == 'string') animate.se = { 1: animate.se };
+
+		var id = setTimeout(null);
+		a.push({
+			"name": name,
+			"id": id,
+			"animate": animate,
+			"centerX": centerX,
+			"centerY": centerY,
+			"start": 0,
+			"pause": false,
+			"pausetime": 0,
+			"index": 0,
+		});
+
+		return id;
+	}
+
+	function animationFrame(callback1) {
+		let time = 0
+		core.registerAnimationFrame("animate2", true, function (timestamp) {
+			let frametime = timestamp - time
+			time = timestamp;
+			if (
+				!a ||
+				a.length == 0
+			)
+				return;
+			core.clearMap(anctx);
+			// 更新帧
+
+			for (var i = 0; i < a.length; i++) {
+				var obj = a[i];
+				if (obj.start === 0) obj.start = time
+				obj.index = Math.floor((time - obj.start) / (1000 / 60))
+				if (obj.index >= obj.animate.frames.length) {
+					(function (callback) {
+						setTimeout(function () {
+							if (callback) callback();
+						});
+					})(obj.callback);
+				}
+			}
+			a = a.filter(function (obj) {
+				return obj.index < obj.animate.frames.length;
+			});
+
+			if (a.length === 0) {
+				core.clearMap(anctx);
+				core.unregisterAnimationFrame("animate2")
+				callback1()
+			}
+			a.forEach(function (obj) {
+				if (obj.hero) {
+					core.maps._drawAnimateFrame(
+						anctx,
+						obj.animate,
+						core.status.heroCenter.px,
+						core.status.heroCenter.py,
+						obj.index
+					);
+				} else {
+					core.maps._drawAnimateFrame(
+						anctx,
+						obj.animate,
+						obj.centerX,
+						obj.centerY,
+						obj.index
+					);
+				}
+			});
+			core.animateFrame.animateTime = timestamp;
+		});
+	}
+
+	function animateonAttack(name, onenemy, callback) {
+		if (onenemy) {
+			drawAnimate(name, 290, 180 + changeY);
+		} else {
+			drawAnimate(name, 130, 180 + changeY);
+		}
+	}
+
+	this.attackAnimate = function (
+		enemyId,
+		heroInfo,
+		enemyInfo,
+		equipInfo,
+		oneTurn,
+		heroDiffList,
+		enemyDiffList,
+		heroanimateList,
+		enemyanimateList
+	) {
+		//参数分别为怪物id、真实属性,战斗信息,特殊装备(如火焰风衣)属性特殊装备属性为以元组{equipId,oneDamage,speed,now:0}构成的数组(列出每个需要计算的特殊装备,没有则为空数组或不填)
+		core.lockControl();
+		core.clearMap(ctx);
+		core.status.event.id = "attackAnimate";
+		let turn = 0;
+		enemyInfo.id = enemyId;
+		enemyInfo.cls = core.getClsFromId(enemyId);
+		enemyInfo.name = core.material.enemys[enemyId].name;
+		let max = heroInfo.speed
+		if (enemyInfo.speed > max) max = enemyInfo.speed
+		equipInfo.forEach(v => { if (v.speed > max) max = v.speed })
+		let i = 1
+		while (oneTurn * i / max < 15) {
+			i++
+		}
+		if (heroInfo.onAttack) heroInfo.now *= i
+		if (enemyInfo.onAttack) enemyInfo.now *= i
+		equipInfo.forEach(v => { if (v.onAttack) v.now *= i })
+		oneTurn *= i
+		let time = 0,
+			farme = 0;
+		return new Promise((res) => {
+			core.plugin.battle_onclick = function (x, y, px, py) {
+				const makeBox = ([x, y], [w, h]) => {
+					return [
+						[x, y],
+						[x + w, y + h],
+					];
+				};
+				const inRect = ([x, y], [
+					[sx, sy],
+					[dx, dy]
+				]) => {
+					return sx <= x && x <= dx && sy <= y && y <= dy;
+				};
+				const pos = [px, py];
+				const easybox = makeBox([90, 232 + changeY], [80, 22]),
+					easyspeedbox = makeBox([290, 232 + changeY], [40, 22]),
+					uneasybox = makeBox([265, 330 + changeY], [65, 20]),
+					uneasyspeedbox = makeBox([290, 350 + changeY], [40, 20]),
+					uneasyClosebox = makeBox([265, 370 + changeY], [65, 20]),
+					easyClosebox = makeBox([290, 255 + changeY], [40, 20]);
+				if (inRect(pos, easybox) && easy) {
+					easy = false;
+				} else if (inRect(pos, uneasybox) && !easy) {
+					easy = true;
+				} else if (
+					(inRect(pos, easyspeedbox) && easy) ||
+					(inRect(pos, uneasyspeedbox) && !easy)
+				) {
+					flags.qukly = !flags.qukly
+
+				} else if ((inRect(pos, easyClosebox) && easy) ||
+					(inRect(pos, uneasyClosebox) && !easy)
+				) {
+					core.status.event.id = "";
+					core.unregisterAnimationFrame("attackAnimate");
+					core.clearMap(ctx);
+					core.closePanel();
+					res();
+				}
+			};
+			async function drawAttackAnimate(
+				heroInfo,
+				oneTurn,
+				enemyInfo,
+				equipInfo,
+				farme,
+				heroDiffList,
+				enemyDiffList,
+				heroanimateList,
+				enemyanimateList
+			) {
+				core.lockControl();
+				core.status.event.id = "battle";
+
+				let attack = false;
+				if (heroInfo.isAttack) attack = true;
+				if (enemyInfo.isAttack) attack = true;
+				equipInfo.forEach(function (v) {
+					if (v.isAttack) attack = true;
+				});
+				let onAttack = false;
+				if (heroInfo.onAttack) onAttack = true;
+				if (enemyInfo.onAttack) onAttack = true;
+				equipInfo.forEach(function (v) {
+					if (v.onAttack) onAttack = true;
+				});
+				core.clearMap(ctx);
+				let animate = Math.floor(farme / 15);
+
+				if (flags.qukly) {
+					while (true) {
+						let goattack = false
+						equipInfo.forEach(v => {
+							if (v.now >= oneTurn) goattack = true
+
+						});
+						if (enemyInfo.now >= oneTurn) goattack = true
+						if (heroInfo.now >= oneTurn) goattack = true
+
+						if (goattack) break;
+						enemyInfo.now += enemyInfo.speed
+						heroInfo.now += heroInfo.speed
+						equipInfo.forEach(function (v) {
+							v.now += v.speed
+
+						});
+
+					}
+				}
+				if (easy) {
+					core.fillRect(ctx, 64, 52, 288, 212, "rgba(0,0,0,0.5)");
+					core.strokeRect(ctx, 64, 52, 288, 212, "rgba(255,255,255,0.5)", 4);
+					core.setTextAlign(ctx, "center");
+					core.fillBoldText(
+						ctx,
+						hero.name,
+						127,
+						148 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Verdana"
+					);
+					core.setTextAlign(ctx, "left");
+					core.drawIcon(ctx, "hp", 70, 210 + changeY, 16, 16);
+					core.fillBoldText(
+						ctx,
+						"生命  " + core.formatBigNumber(heroInfo.hp, true),
+						90,
+						225 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+
+					core.fillBoldText(
+						ctx,
+						"详细模式",
+						90,
+						250 + changeY,
+						"#FFFF60",
+						"#000000",
+						"bold 18px Verdana"
+					);
+
+					core.strokeRect(ctx, 112, 159 + changeY, 32, 48, "rgba(255,255,255,1)", 1);
+					let img =
+						attack && heroDiffList[turn].hp < 0 ?
+						imagelighter(core.material.images.images["hero.webp"]) :
+						attack && heroDiffList[turn].hp > 0 ?
+						imagelighter(
+							core.material.images.images["hero.webp"],
+							"rgba(0, 255, 0, 0.5)"
+						) :
+						core.material.images.images["hero.webp"];
+					core.drawImage(
+						ctx,
+						img,
+						32 * (animate % 4),
+						0,
+						32,
+						48,
+						112,
+						159 + changeY,
+						32,
+						48
+					);
+
+					core.setTextAlign(ctx, "center");
+					core.fillBoldText(
+						ctx,
+						enemyInfo.name,
+						289,
+						148 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Verdana"
+					);
+
+					core.setTextAlign(ctx, "right");
+					if (enemyInfo.cls === "enemys") {
+						core.strokeRect(ctx, 272, 175 + changeY, 32, 32, "rgba(255,255,255,1)", 1);
+						let img =
+							attack && enemyDiffList[turn].hp < 0 ?
+							imagelighter(core.getBlockInfo(enemyInfo.id).image) :
+							attack && enemyDiffList[turn].hp > 0 ?
+							imagelighter(
+								core.getBlockInfo(enemyInfo.id).image,
+								"rgba(0, 255, 0, 0.5)"
+							) :
+							core.getBlockInfo(enemyInfo.id).image;
+						core.drawImage(
+							ctx,
+							img,
+							32 * (animate % 2),
+							core.getBlockInfo(enemyInfo.id).posY * 32,
+							32,
+							32,
+							272,
+							175 + changeY,
+							32,
+							32
+						);
+					} else {
+						core.strokeRect(ctx, 272, 159 + changeY, 32, 48, "rgba(255,255,255,1)", 1);
+						let img =
+							attack && enemyDiffList[turn].hp < 0 ?
+							imagelighter(core.getBlockInfo(enemyInfo.id).image) :
+							attack && enemyDiffList[turn].hp > 0 ?
+							imagelighter(
+								core.getBlockInfo(enemyInfo.id).image,
+								"rgba(0, 255, 0, 0.5)"
+							) :
+							core.getBlockInfo(enemyInfo.id).image;
+						core.drawImage(
+							ctx,
+							img,
+							32 * (animate % 4),
+							core.getBlockInfo(enemyInfo.id).posY * 48,
+							32,
+							48,
+							272,
+							159 + changeY,
+							32,
+							48
+						);
+					}
+					core.drawIcon(ctx, "hp", 330, 210 + changeY, 16, 16);
+
+					core.fillBoldText(
+						ctx,
+						core.formatBigNumber(enemyInfo.hp, true) + "  生命",
+						330,
+						225 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Verdana"
+					);
+					core.fillBoldText(
+						ctx,
+						flags.qukly ? "正常" : "极速",
+						330,
+						250 + changeY,
+						"#FFFF60",
+						"#000000",
+						"bold 18px Verdana"
+					);
+					core.fillBoldText(
+						ctx,
+						"跳过",
+						330,
+						273 + changeY,
+						"#FFFF60",
+						"#000000",
+						"bold 18px Verdana"
+					);
+					core.fillBoldText(
+						ctx,
+						"V",
+						219,
+						183 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 48px pala"
+					);
+					core.fillBoldText(
+						ctx,
+						"s",
+						231,
+						183 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 36px pala"
+					);
+
+					if (!attack && !onAttack && !flags.qukly) enemyInfo.now += enemyInfo.speed;
+					let enemynow = Math.min(100 + (enemyInfo.now / oneTurn) * 215, 315);
+					ctx.fillStyle = "#FFFFFF";
+					ctx.beginPath();
+					ctx.moveTo(enemynow, 120 + changeY);
+					ctx.lineTo(enemynow + 5, 110 + changeY);
+					ctx.lineTo(enemynow - 5, 110 + changeY);
+					ctx.closePath();
+					ctx.fill();
+					core.drawIcon(ctx, 'T397', enemynow - 16, 84 + changeY, 32, 32)
+
+					core.drawLine(ctx, 100, 125 + changeY, 315, 125 + changeY, "#FFFFFF", 5);
+					equipInfo.forEach(function (v) {
+						if (!attack && !onAttack) v.now += v.speed;
+						let vnow = Math.min(100 + (v.now / oneTurn) * 215, 315);
+						ctx.beginPath();
+						ctx.moveTo(vnow, 120 + changeY);
+						ctx.lineTo(vnow + 5, 110 + changeY);
+						ctx.lineTo(vnow - 5, 110 + changeY);
+						ctx.closePath();
+
+						ctx.fill();
+
+						core.drawIcon(ctx, v.id, vnow - 16, 84 + changeY, 32, 32);
+					});
+					if (!attack && !onAttack && !flags.qukly) heroInfo.now += hero.speed;
+					let heronow = Math.min(100 + (heroInfo.now / oneTurn) * 215, 315);
+					ctx.beginPath();
+					ctx.moveTo(heronow, 120);
+					ctx.lineTo(heronow + 5, 110);
+					ctx.lineTo(heronow - 5, 110);
+					ctx.closePath();
+
+					ctx.fill();
+					core.drawIcon(ctx, 'T399', heronow - 16 - 4, 84 - 4 + changeY, 40, 40)
+
+				} else {
+					core.fillRect(ctx, 64, 52, 288, 320, "rgba(0,0,0,0.5)");
+					core.strokeRect(ctx, 64, 52, 288, 320, "rgba(255,255,255,0.5)", 4);
+					core.setTextAlign(ctx, "center");
+					core.fillBoldText(
+						ctx,
+						hero.name,
+						127,
+						148 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Verdana"
+					);
+					core.setTextAlign(ctx, "left");
+					core.drawIcon(ctx, "hp", 70, 210 + changeY, 16, 16);
+					core.drawIcon(ctx, "atk", 70, 230 + changeY, 16, 16);
+					core.drawIcon(ctx, "def", 70, 250 + changeY, 16, 16);
+					core.drawIcon(ctx, "I374", 70, 270 + changeY, 16, 16);
+					core.drawIcon(ctx, "I375", 70, 290 + changeY, 16, 16);
+					core.drawIcon(ctx, "mdef", 70, 310 + changeY, 16, 16);
+					core.drawIcon(ctx, "amulet", 70, 330 + changeY, 16, 16);
+					core.drawIcon(ctx, "jumpShoes", 70, 350 + changeY, 16, 16);
+					core.fillBoldText(
+						ctx,
+						"生命  " + core.formatBigNumber(heroInfo.hp, true),
+						90,
+						225 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+
+					core.fillBoldText(
+						ctx,
+						"攻击  " + core.formatBigNumber(heroInfo.atk),
+						90,
+						245 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						"防御  " + core.formatBigNumber(heroInfo.def),
+						90,
+						265 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						"法强  " + core.formatBigNumber(heroInfo.spell),
+						90,
+						285 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						"法攻  " + core.formatBigNumber(heroInfo.matk),
+						90,
+						305 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						"护盾  " + core.formatBigNumber(heroInfo.mhp),
+						90,
+						325 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						"法抗  " + heroInfo.mdef + "%",
+						90,
+						345 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						"速度  " + core.formatBigNumber(heroInfo.speed),
+						90,
+						365 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+
+					core.strokeRect(ctx, 112, 159 + changeY, 32, 48, "rgba(255,255,255,1)", 1);
+					let img =
+						attack && heroDiffList[turn].hp < 0 ?
+						imagelighter(core.material.images.images["hero.webp"]) :
+						attack && heroDiffList[turn].hp > 0 ?
+						imagelighter(
+							core.material.images.images["hero.webp"],
+							"rgba(0, 255, 0, 0.5)"
+						) :
+						core.material.images.images["hero.webp"];
+					core.drawImage(
+						ctx,
+						img,
+						32 * (animate % 4),
+						0,
+						32,
+						48,
+						112,
+						159 + changeY,
+						32,
+						48
+					);
+
+					core.setTextAlign(ctx, "center");
+					core.fillBoldText(
+						ctx,
+						enemyInfo.name,
+						289,
+						148 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Verdana"
+					);
+
+					core.setTextAlign(ctx, "right");
+					if (enemyInfo.cls === "enemys") {
+						core.strokeRect(ctx, 272, 175 + changeY, 32, 32, "rgba(255,255,255,1)", 1);
+						let img =
+							attack && enemyDiffList[turn].hp < 0 ?
+							imagelighter(core.getBlockInfo(enemyInfo.id).image) :
+							attack && enemyDiffList[turn].hp > 0 ?
+							imagelighter(
+								core.getBlockInfo(enemyInfo.id).image,
+								"rgba(0, 255, 0, 0.5)"
+							) :
+							core.getBlockInfo(enemyInfo.id).image;
+						core.drawImage(
+							ctx,
+							img,
+							32 * (animate % 2),
+							core.getBlockInfo(enemyInfo.id).posY * 32,
+							32,
+							32,
+							272,
+							175 + changeY,
+							32,
+							32
+						);
+					} else {
+						core.strokeRect(ctx, 272, 159 + changeY, 32, 48, "rgba(255,255,255,1)", 1);
+						let img =
+							attack && enemyDiffList[turn].hp < 0 ?
+							imagelighter(core.getBlockInfo(enemyInfo.id).image) :
+							attack && enemyDiffList[turn].hp > 0 ?
+							imagelighter(
+								core.getBlockInfo(enemyInfo.id).image,
+								"rgba(0, 255, 0, 0.5)"
+							) :
+							core.getBlockInfo(enemyInfo.id).image;
+						core.drawImage(
+							ctx,
+							img,
+							32 * (animate % 4),
+							core.getBlockInfo(enemyInfo.id).posY * 48,
+							32,
+							48,
+							272,
+							159 + changeY,
+							32,
+							48
+						);
+					}
+					core.drawIcon(ctx, "hp", 330, 210 + changeY, 16, 16);
+					core.drawIcon(ctx, "atk", 330, 230 + changeY, 16, 16);
+					core.drawIcon(ctx, "def", 330, 250 + changeY, 16, 16);
+					core.drawIcon(ctx, "I374", 330, 270 + changeY, 16, 16);
+					core.drawIcon(ctx, "amulet", 330, 290 + changeY, 16, 16);
+					core.drawIcon(ctx, "jumpShoes", 330, 310 + changeY, 16, 16);
+					core.fillBoldText(
+						ctx,
+						core.formatBigNumber(enemyInfo.hp, true) + "  生命",
+						330,
+						225 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						core.formatBigNumber(enemyInfo.atk) + "  攻击",
+						330,
+						245 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						core.formatBigNumber(enemyInfo.def) + "  防御",
+						330,
+						265 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						(enemyInfo.spell ?? 0) + "  法强",
+						330,
+						285 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						(enemyInfo.mdef ?? 0) * 100 + "%  法抗",
+						330,
+						305 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						core.formatBigNumber(enemyInfo.speed) + "  速度",
+						330,
+						325 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 14px Arial"
+					);
+					core.fillBoldText(
+						ctx,
+						"简易模式",
+						330,
+						345 + changeY,
+						"#FFFF60",
+						"#000000",
+						"bold 16px Verdana"
+					);
+					core.fillBoldText(
+						ctx,
+						flags.qukly ? "正常" : "极速",
+						330,
+						365 + changeY,
+						"#FFFF60",
+						"#000000",
+						"bold 16px Verdana"
+					);
+					core.fillBoldText(
+						ctx,
+						"跳过",
+						330,
+						385 + changeY,
+						"#FFFF60",
+						"#000000",
+						"bold 16px Verdana"
+					);
+					core.fillBoldText(
+						ctx,
+						"V",
+						219,
+						183 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 48px pala"
+					);
+					core.fillBoldText(
+						ctx,
+						"s",
+						231,
+						183 + changeY,
+						"#FFFFFF",
+						"#000000",
+						"bold 36px pala"
+					);
+
+					if (!attack && !onAttack & !flags.qukly) enemyInfo.now += enemyInfo.speed;
+					let enemynow = Math.min(100 + (enemyInfo.now / oneTurn) * 215, 315);
+					ctx.fillStyle = "#FFFFFF";
+					ctx.beginPath();
+					ctx.moveTo(enemynow, 120 + changeY);
+					ctx.lineTo(enemynow + 5, 110 + changeY);
+					ctx.lineTo(enemynow - 5, 110 + changeY);
+					ctx.closePath();
+					ctx.fill();
+					core.drawIcon(ctx, 'T397', enemynow - 16, 84 + changeY, 32, 32)
+
+					core.drawLine(ctx, 100, 125 + changeY, 315, 125 + changeY, "#FFFFFF", 5);
+					equipInfo.forEach(function (v) {
+						if (!attack && !onAttack) v.now += v.speed;
+						let vnow = Math.min(100 + (v.now / oneTurn) * 215, 315);
+						ctx.beginPath();
+						ctx.moveTo(vnow, 120 + changeY);
+						ctx.lineTo(vnow + 5, 110 + changeY);
+						ctx.lineTo(vnow - 5, 110 + changeY);
+						ctx.closePath();
+
+						ctx.fill();
+
+						core.drawIcon(ctx, v.id, vnow - 16, 84 + changeY, 32, 32);
+					});
+					if (!attack && !onAttack && !flags.qukly) heroInfo.now += hero.speed;
+					let heronow = Math.min(100 + (heroInfo.now / oneTurn) * 215, 315);
+					ctx.beginPath();
+					ctx.moveTo(heronow, 120 + changeY);
+					ctx.lineTo(heronow + 5, 110 + changeY);
+					ctx.lineTo(heronow - 5, 110 + changeY);
+					ctx.closePath();
+
+					ctx.fill();
+					core.drawIcon(ctx, 'T399', heronow - 16 - 4, 84 - 4 + changeY, 40, 40)
+				}
+				let nowattacking = false;
+				if (heroInfo.now >= oneTurn && !heroInfo.isAttack) {
+					heroInfo.onAttack = false;
+					heroInfo.isAttack = true;
+					nowattacking = true;
+				}
+				if (enemyInfo.now >= oneTurn && !enemyInfo.isAttack) {
+					enemyInfo.onAttack = false;
+					enemyInfo.isAttack = true;
+					nowattacking = true;
+				}
+				const equipanimate = [];
+				equipInfo.forEach((v) => {
+					if (v.now >= oneTurn && !v.isAttack) {
+						v.isAttack = true;
+						v.onAttack = false;
+						nowattacking = true;
+						equipanimate.push(v);
+					}
+				});
+				if (!attack && nowattacking) {
+					let herodamage = enemyDiffList[turn].hp;
+					if (herodamage > 0) herodamage = "+" + herodamage;
+					let text = herodamage === 0 ? "抵抗" : herodamage;
+
+					Dove.MorePerform.ShowDamagePop.PopDamage(
+						ctx, // 默认画布名称
+						270, // 英雄位置 x
+						160 + changeY, // 英雄位置 y
+						text, // 伤害值
+						18, // 默认字体大小
+						"Arial", //默认字体
+						typeof text === "string" && text.startsWith("+") ?
+						"#22FF44" :
+						typeof text === "string" ?
+						"#FFFFFF" :
+						null, // 默认颜色
+						text === "抵抗" ? "#000000" : null, // 默认描边颜色
+						0, // 默认水平速度
+						-1, // 默认垂直速度
+						0, // 默认重力
+						90 // 默认显示时长(帧数)
+					);
+					for (const v in enemyDiffList[turn]) {
+						enemyInfo[v] += enemyDiffList[turn][v];
+					}
+					enemyanimateList[turn].forEach((v) => animateonAttack(v, true));
+					let enemydamage = heroDiffList[turn].hp;
+					if (enemydamage > 0) enemydamage = "+" + enemydamage;
+					text = enemydamage === 0 ? "抵抗" : enemydamage;
+					Dove.MorePerform.ShowDamagePop.PopDamage(
+						ctx, // 默认画布名称
+						110, // 英雄位置 x
+						160 + changeY, // 英雄位置 y
+						text, // 伤害值
+						18, // 默认字体大小
+						"Arial", //默认字体
+						typeof text === "string" && text.startsWith("+") ?
+						"#22FF44" :
+						typeof text === "string" ?
+						"#FFFFFF" :
+						null, // 默认颜色
+						text === "抵抗" ? "#000000" : null, // 默认描边颜色
+						0, // 默认水平速度
+						-1, // 默认垂直速度
+						0, // 默认重力
+						90 // 默认显示时长(帧数)
+					);
+					for (const v in heroDiffList[turn]) {
+						heroInfo[v] += heroDiffList[turn][v];
+					}
+					heroanimateList[turn].forEach((v) => animateonAttack(v, false));
+
+					if (enemyDiffList[turn] < 0) enemyInfo.inAttack = true;
+					if (heroDiffList[turn] < 0) heroInfo.inAttack = true;
+					if (heroInfo.hp < 0) heroInfo.hp = 0;
+					if (enemyInfo.hp < 0) enemyInfo.hp = 0;
+					await Promise.all([
+
+						await new Promise((resolve) => animationFrame(resolve)),
+						new Promise((resolve) => {
+							if (heroInfo.isAttack) {
+								heroInfo.now = 0;
+								heroInfo.isAttack = false;
+							}
+							resolve();
+						}),
+						new Promise((resolve) => {
+							if (enemyInfo.isAttack) {
+								enemyInfo.now = 0;
+								enemyInfo.isAttack = false;
+							}
+							resolve();
+						}),
+						new Promise((resolve) => {
+							if (equipanimate.length > 0) {
+								equipanimate.forEach((v) => {
+									v.now = 0;
+									v.isAttack = false;
+								});
+							}
+							resolve();
+						}),
+						new Promise((resolve) => {
+							turn++;
+							resolve();
+						}),
+					]);
+
+					if (heroInfo.hp <= 0 || enemyInfo.hp <= 0) {
+						core.status.event.id = "";
+						core.unregisterAnimationFrame("attackAnimate");
+						core.clearMap(ctx);
+						core.closePanel();
+						res();
+					}
+				}
+			}
+
+			core.registerAnimationFrame("attackAnimate", true, (temptime) => {
+				if (!time) time = temptime
+				farme = Math.floor((temptime - time) / (1000 / 60))
+				time = temptime;
+				drawAttackAnimate(
+					heroInfo,
+					oneTurn,
+					enemyInfo,
+					equipInfo,
+					farme,
+					heroDiffList,
+					enemyDiffList,
+					heroanimateList,
+					enemyanimateList
+				);
+
+
+			});
+		});
+	};
+},
+    "剧情内容": function () {
+	// 每项为一个数组,第一项是名字,第二项是对话内容,第三项为音频文件名(没有则不需要第三项)
+	// 回放只会在同一个this下回放,进入剧情前请以事件块声明进入哪个剧情数组
+	this.chapter000 = [
+		["", "这些天,街道不曾下雨。"],
+		["", "所以,那浸湿地面的,定是那些女孩们流落的鲜血无疑。"],
+		["", "我蹲在充斥着铁锈味般恶臭的小巷中,悠闲地如是想着。"],
+		["", "扑哧。"],
+		["", "耳旁再次响起象征着某个女孩子死去的声音。"],
+		["", "再一次——"],
+		["", "再一次。"],
+		["", "女子们被肢解成单纯的肉块。"],
+		["", "我任由流下的血浸满全身,屏住自己的呼吸。"],
+		["", "祈求自己能拥有从猎人手中逃脱的幸运。"],
+		["", "扑哧。"],
+		["", "直到刚才,我们还坐在去往娼馆的马车的路上。"],
+		["", "而在这之中的某些人,已经不在这个世上了。"],
+		["", "不,应该把“某些”换成“几乎所有”才更为恰当吧。"],
+		["", "恐怕,不久之后我也会变成小巷中血腥的装饰品。"],
+		["", "我是为了得到这种死法,才辛苦苟活至今的吗?"],
+		["", "来个人告诉我啊——"],
+		["", "谁都好。"],
+		["", "来人啊!!"],
+		["少女", "「呃······!?」", "aiy010000010.opus"], //小动物01
+		["", "漆黑的物体充斥了我的整个视野"],
+		["", "我很快意识到,那是只很大的脚。"],
+		["", "必须要出声求救。"],
+		["", "可是,耳中却只能听到自己的牙关不停交战的声音。"],
+		["", "我是如此的无助。"],
+		["", "逃跑也好,道歉也罢。"],
+		["", "就连抬头看一眼将要杀掉我的人的面孔都做不到。"],
+		["少女", "「······被杀」", "aiy010000020.opus"], //小动物02
+		["", "会被杀。"],
+		["", "会被杀!!"],
+		["", "来自内心深处的冰冷预感,渐渐地在体内蔓延开来。"],
+		["少女", "「不,不要······」", "aiy010000030.opus"], //小动物03
+	]
+	this.chapter001 = [
+		["", "浮游都市,《诺瓦斯·艾蒂尔》。"],
+		["", "《特别受灾地区》——"],
+		["", "通称,《牢狱》"],
+		["", "是被险峻的峭壁环绕,与世隔绝的,都市的最底部。"],
+	]
+	this.chapter002 = [
+		["年轻人", "「放开我!」", "aiy710000010.opus"], //龙套1-01
+		["年轻人", "「我只是在帮那个女人而已!」", "aiy710000020.opus"], //龙套1-02
+		["年轻人", "「你们没听到吗!?」", "aiy710000030.opus"], //龙套1-03
+		["年轻人", "「她是被受骗才会被卖到娼馆来的」", "aiy710000040.opus"], //龙套1-04
+		[
+			"年轻人",
+			"「用肮脏的手段把钱借给她父母的,就是你们这些家伙吧!?」",
+			"aiy710000050.opus",
+		], //龙套1-05
+		["年轻人", "「给我说些什么啊」", "aiy710000060.opus"], //龙套1-06
+		["凯伊姆", "「这些话等到了娼馆再说吧」", "aiy310000010.opus"], //男主01
+		["凯伊姆", "「我来抓你,只是受雇于人而已」", "aiy310000020.opus"], //男主02
+		["", "我走进娼馆《莉莉乌姆》的接待室。"],
+		["", "正在桌旁整理账簿的奥兹停下手头的工作,抬起头向我看来。"],
+		["奥兹", "「这不是凯伊姆先生吗,辛苦了」", "aiy350000010.opus"], //金锁高官01
+		["奥兹", "「委托已经完成了吗?」", "aiy350000020.opus"], //金锁高官02
+		["凯伊姆", "「啊啊,是这家伙没错吧」", "aiy310000030.opus"], //男主03
+		["", "奥兹用只要接触到就能杀人般的眼神在男人脸上搜过。"],
+		["奥兹", "「没错,就是这个人」", "aiy350000030.opus"], //金锁高官03
+		["凯伊姆", "「是么」", "aiy310000040.opus"], //男主04
+		["年轻人", "「你,你们要对我做什么」", "aiy710000070.opus"], //龙套1-07
+		["奥兹", "「······」", "aiy350000040.opus"], //金锁高官04
+		["", "奥兹用一个眼神,就让男人闭上了嘴。"],
+		["", "然后,向我这边转过身来。"],
+		["奥兹", "「抱歉啊,总是麻烦你去做这些无聊的事」", "aiy350000050.opus"], //金锁高官05
+		["奥兹", "「都怪我们这边的年轻人太没用」", "aiy350000060.opus"], //金锁高官06
+		["凯伊姆", "「客套话就免了」", "aiy310000050.opus"], //男主05
+		["奥兹", "「这还真是失礼了」", "aiy350000070.opus"], //金锁高官07
+		["奥兹", "「喂,来个人」", "aiy350000080.opus"], //金锁高官08
+		["光头男人", "「是」", "aiy820000010.opus"], //龙套2-01
+		["奥兹", "「凯伊姆先生做完工作回来了」", "aiy350000090.opus"], //金锁高官09
+		["光头男人", "「是,是,那个······」", "aiy820000020.opus"], //龙套2-02
+		["奥兹", "「我是要你拿些酒来,这个蠢材!」", "aiy350000100.opus"], //金锁高官10
+		["", "喀!"],
+		["", "奥兹扔出的烟灰缸砸中了手下的额头。"],
+		["", "鲜血四溅。"],
+		["凯伊姆", "「不用这么麻烦」", "aiy310000060.opus"], //男主06
+		["凯伊姆", "「我接下来要去《菲诺列塔》」", "aiy310000070.opus"], //男主07
+		["奥兹", "「喔唷」", "aiy350000110.opus"], //金锁高官11
+		[
+			"奥兹",
+			"「既然如此,我就不留您在这里喝难饮的劣质酒了」",
+			"aiy350000120.opus",
+		], //金锁高官12
+		["", "奥兹斜眼看着正捂住额头呻吟的手下,轻描淡写地说道。"],
+		["凯伊姆", "「用这些钱去买药」", "aiy310000080.opus"], //男主08
+		["", "我将几枚铜钱仍在那个手下的身前。"],
+		["奥兹", "「凯伊姆先生,不用对他们这么好」", "aiy350000130.opus"], //金锁高官13
+		["凯伊姆", "「无妨」", "aiy310000090.opus"], //男主09
+		["凯伊姆", "「话说回来,那个要落跑的女人呢?」", "aiy310000100.opus"], //男主10
+		[
+			"奥兹",
+			"「我把她交给那些年轻人了,现在应该正在体会人生的严苛吧」",
+			"aiy350000140.opus",
+		], //金锁高官14
+		[
+			"奥兹",
+			"「正好,趁此机会凯伊姆先生也来享受一番如何?」",
+			"aiy350000150.opus",
+		], //金锁高官15
+		["年轻人", "「你,你们这些家伙,要对她做什么!?」", "aiy710000080.opus"], //龙套1-08
+		["", "咣!"],
+		["", "奥兹给了他一拳。"],
+		["", "一击即倒。"],
+		["", "喀,咚,咯!"],
+		["", "奥兹毫不留情地向男人的脸上踩去。"],
+		["年轻人", "「咕······呃咳······」", "aiy710000090.opus"], //龙套1-09
+		["", "折断的牙齿伴着血泡被吐出。"],
+		["", "这份白色在鲜红色的液体中格外显眼。"],
+		[
+			"年轻人",
+			"「你们以为做出这种事······卫兵会坐视不理吗······」",
+			"aiy710000100.opus",
+		], //龙套1-10
+		["奥兹", "「啊啊,不会坐视不理的」", "aiy350000160.opus"], //金锁高官16
+		[
+			"奥兹",
+			"「应该会拿出你的钱包,和我们商量如何瓜分吧」",
+			"aiy350000170.opus",
+		], //金锁高官17
+		["年轻人", "「那,那种事······」", "aiy710000110.opus"], //龙套1-11
+		["", "这在牢狱是理所当然的事。"],
+		["奥兹", "「怎么,头一回来牢狱么?」", "aiy350000180.opus"], //金锁高官18
+		["", "男人点了点头。"],
+		[
+			"奥兹",
+			"「为了被骗的女人而来到牢狱,真是个规矩人啊」",
+			"aiy350000190.opus",
+		], //金锁高官19
+		["奥兹", "「······前提是,被骗的人不是你」", "aiy350000200.opus"], //金锁高官20
+		["年轻人", "「你说······我被骗了?」", "aiy710000120.opus"], //龙套1-12
+		["年轻人", "「那,那是怎么回事!?」", "aiy710000130.opus"], //龙套1-13
+		["奥兹", "「不用急,今天晚上会好好告诉你的」", "aiy350000210.opus"], //金锁高官21
+		["", "奥兹抓起男人的脸。"],
+		["", "为引诱客人的怜悯之心而装纯,是娼妇的惯用手段。"],
+		["", "双亲被骗而借钱,结果作为抵押而将自己卖到这里,这是典型的说法。"],
+		[
+			"",
+			"如果只是头脑发热而成为常客也就罢了,这次的男人热血过头,居然想出了要带女人私奔的计划。",
+		],
+		[
+			"",
+			"虽然女人半开玩笑地予以拒绝,但不知天高地厚的这家伙还是拉着她逃跑了。",
+		],
+		["", "不过,想要逃脱追击本来就是不可能的任务。"],
+		["", "但即便如此,这种事情还是会一再的出现。"],
+		["", "说谎的女人和被骗的男人。"],
+		["", "在娼馆街,这是令人看到生厌的日常的风景。"],
+		["凯伊姆", "「我要走了」", "aiy310000110.opus"], //男主11
+		["奥兹", "「好的,下次再麻烦您」", "aiy350000220.opus"], //金锁高官22
+		["奥兹", "「之后吉克先生会将谢礼交给您的」", "aiy350000230.opus"], //金锁高官23
+		["凯伊姆", "「啊啊」", "aiy310000120.opus"], //男主12
+		["", "我背向奥兹走出娼馆。"],
+		["凯伊姆", "「······?」", "aiy310000130.opus"], //男主13
+		["", "从远方传来微弱的歌声。"],
+		["", "是关卡广场的方向。"],
+		["", "对了。"],
+		["", "今天有觐见圣女的仪式。"],
+		["", "当代的圣女伊莲——"],
+		["", "俗称《盲眼之圣女》,据说即使在历代的圣女中,人气也是数一数二的。"],
+		["", "广场上的人估计相当多吧。"],
+		["", "虽然我也想去看看她长什么样,不过要在人潮中挤来挤去就免了。"],
+		["", "还是老老实实去菲诺列塔喝烧酒吧。"],
+		["", "正当我这样想着的时候,一个身影自小巷的那头走来。"],
+		["凯伊姆", "「艾莉斯」", "aiy310000140.opus"], //男主14
+	];
+	this.chapter01 = [
+		["艾莉斯", "「啊,凯伊姆」", "aiy020000005.opus"], //医生00.5
+		["艾莉斯", "「正好,我还想要去找你呢」", "aiy020000010.opus"], //医生01
+		[
+			"艾莉斯",
+			"「没想到凯伊姆会主动出现······这是命运吗?」",
+			"aiy020000020.opus",
+		], //医生02
+		["凯伊姆", "「显然不是吧」", "aiy310000150.opus"], //男主15
+		["艾莉斯", "「啊,是么」", "aiy020000030.opus"], //医生03
+		["", "艾莉斯挑了挑整齐的双眉,微微地哼了一声。"],
+		[
+			"",
+			"虽然是个相当引人注目的美人,但她这个将亲切儿子丢入无底深渊的性格,为自己扣了不少的分",
+		],
+		["", "给人印象最深的,就是那潭水般的双瞳。"],
+		["", "在漆黑的瞳孔中,完全看不出感情的波动。"],
+		["艾莉斯", "「喜欢我的眼睛吗?」", "aiy020000040.opus"], //医生04
+		["艾莉斯", "「如果想要的话就给你吧?」", "aiy020000050.opus"], //医生05
+		["凯伊姆", "「用不着」", "aiy310000160.opus"], //男主16
+		["艾莉斯", "「阿拉,可惜」", "aiy020000060.opus"], //医生06
+		["凯伊姆", "「那么,找我有什么事」", "aiy310000170.opus"], //男主17
+		["艾莉斯", "「梅尔特的钱好像被偷了」", "aiy020000070.opus"], //医生07
+		["凯伊姆", "「钱被偷了?都几岁了还这么没用」", "aiy310000180.opus"], //男主18
+		["艾莉斯", "「不要对我说啊」", "aiy020000080.opus"], //医生08
+		[
+			"凯伊姆",
+			"「那家伙,该不会说要让我去抓那个小偷吧?」",
+			"aiy310000190.opus",
+		], //男主19
+		["艾莉斯", "「就是这样」", "aiy020000090.opus"], //医生09
+		["凯伊姆", "「笨蛋吗」", "aiy310000200.opus"], //男主20
+		["凯伊姆", "「如果是小钱的话,就当做是买个教训吧」", "aiy310000210.opus"], //男主21
+		["艾莉斯", "「说起来,被盗的是这个月的上纳金」", "aiy020000100.opus"], //医生10
+		["凯伊姆", "「你说什么?」", "aiy310000220.opus"], //男主22
+		["艾莉斯", "「用这些钱买教训,也太过奢侈了呢」", "aiy020000110.opus"], //医生11
+		["凯伊姆", "「知道了,我去找」", "aiy310000230.opus"], //男主23
+		["凯伊姆", "「小偷的特征呢」", "aiy310000240.opus"], //男主24
+		["艾莉斯", "「男孩子」", "aiy020000120.opus"], //医生12
+		["艾莉斯", "「······而且,背后有翅膀」", "aiy020000130.opus"], //医生13
+		[
+			"艾莉斯",
+			"「虽然姑且是藏在身后,但是仔细观察的话是很明显的」",
+			"aiy020000140.opus",
+		], //医生14
+		["凯伊姆", "「羽化病吗」", "aiy310000250.opus"], //男主25
+		[
+			"艾莉斯",
+			"「那些人可是毫不留情的,所以即使是为了那个孩子,也要赶快抓到他」",
+			"aiy020000150.opus",
+		], //医生15
+		["凯伊姆", "「注意到他逃窜的方向了吗?」", "aiy310000260.opus"], //男主26
+		["艾莉斯", "「广场那边」", "aiy020000160.opus"], //医生16
+		[
+			"艾莉斯",
+			"「虽然刚才《不蚀金锁》的人去追了,不过多半是······」",
+			"aiy020000170.opus",
+		], //医生17
+		["凯伊姆", "「偏偏还是广场吗」", "aiy310000280.opus"], //男主28
+		["艾莉斯", "「今天是觐见圣女大人的日子」", "aiy020000180.opus"], //医生18
+		["凯伊姆", "「我知道」", "aiy310000290.opus"], //男主29
+		["凯伊姆", "「尽量找找看就好」", "aiy310000300.opus"], //男主30
+	];
+	this.chapter02 = [
+		["不蚀金锁成员", "「凯伊姆先生,凯伊姆先生」", "aiy860000010.ogg"],
+		[
+			"不蚀金锁成员",
+			"「您已经和艾莉斯大夫见过面了吗?」",
+			"aiy860000020.ogg",
+		],
+		["凯伊姆", "「啊啊,所以才会追过来的」", "aiy310000310.ogg"], //男主31
+		["凯伊姆", "「看到小偷了吗?」", "aiy310000320.ogg"], //男主32
+		[
+			"不蚀金锁成员",
+			"「没有,他向广场那边逃了过去,今天这么拥挤,我们也只能放弃了」",
+			"aiy860000030.ogg",
+		],
+		[
+			"不蚀金锁成员",
+			"「不过,我也只是刚好在店里所以才追了过去,并不是受人所托」",
+			"aiy860000040.ogg",
+		],
+		["不蚀金锁成员", "「我已经准备撤退了」", "aiy860000050.ogg"],
+		["不蚀金锁成员", "「凯伊姆先生还要继续追吗?」", "aiy860000060.ogg"],
+		["凯伊姆", "「啊啊」", "aiy310000330.ogg"], //男主33
+		["", "做完情报交换之后,我跟男人道别。"],
+		["凯伊姆", "「和我想的一样啊······」", "aiy310000340.ogg"], //男主34
+		["", "在牢狱中最大的广场上,聚集着看不到尽头的人群。"],
+		["", "就算是来参见圣女祈祷,这人数也太多了点吧。"],
+		["", "自然,我也找不到逃跑的孩子。"],
+		["", "是混杂到人群中了吧。"],
+		["", "如果已经从广场上脱身了的话,就更难发现了。"],
+		["", "只好赌他还在这里了。"],
+		["", "我先移动到了一个视野良好的地方。"],
+		["", "从这里,一眼就可以看到人群的变化。"],
+		["", "广场还是沸腾起来。"],
+		["", "抬头望去,原来是在天台之上出现了一个人影"],
+		["", "但是,与周围的期待不同,现身的是一名中年的神官。"],
+		["", "骂声四溢。"],
+		["", "神官则是笑着摆正衣领"],
+		["神官", "「从现在开始,举行谒见的仪式」", "aiy440000010.ogg"], //神官01
+		[
+			"神官",
+			"「在参见那位大人之前,我希望牢狱的诸位再次思考这个《诺瓦斯·艾蒂尔》存在的意义······」",
+			"aiy440000020.ogg",
+		], //神官02
+		[
+			"神官",
+			"「初代圣女伊莲大人,便是也难怪这崇高的祈祷之力,令《诺瓦斯·艾蒂尔》浮在空中,拯救了我们的祖先」",
+			"aiy440000030.ogg",
+		], //神官03
+		[
+			"神官",
+			"「这之后的几百年来,传承了初代大人力量的历代圣女伊莲大人,让这里留在了空中」",
+			"aiy440000040.ogg",
+		], //神官04
+		[
+			"神官",
+			"「这座都市是被圣女大人守护的人类最后的圣域,而我们则是被选召的虔诚的信徒」",
+			"aiy440000050.ogg",
+		], //神官05
+		[
+			"神官",
+			"「怀着对圣女的感激祈祷吧,感谢圣女伊莲吧!并献上祈祷!」",
+			"aiy440000060.ogg",
+		], //神官06
+		["圣女", "「不忘感谢与祈祷,神才会拯救我们」", "aiy030000010.ogg"], //圣女01
+		["圣女", "「与我一起,向神虔诚地祈祷吧」", "aiy030000020.ogg"], //圣女02
+		["", "广场上欢声雷动。"],
+		["", "圣女没有回应喧嚣的人声,而是静静地合上双眼面向广场。"],
+		["", "虽然感觉有些冷淡,但总比像个傻瓜似的笑着向这群人挥手要强。"],
+		["", "她掌握着这条街道,还有在这条街上生活的人的命运。"],
+		["", "比起揽得人气,她更想要为了街道的继续存在而献出全力。"],
+		["", "也是为了不让《大崩落》的惨剧再度发生。"],
+		["", "十几年前的那场悲剧。"],
+		[
+			"",
+			"虽然在我脑海中的记忆已经相当模糊,但哪怕只是稍有触及,不快的感觉都会在胸口蔓延开。",
+		],
+		["凯伊姆", "「······」", "aiy310000350.ogg"], //男主35
+		["", "这时我才想起,现在不是我在这里看圣女的时候。"],
+		["女声", "「——っ!?」", "aiy510000010.ogg"],
+		["围观的女人", "「羽,羽化病人!?」", "aiy510000020.ogg"],
+		["围观的中年人", "「喂,谁去叫下羽狩」", "aiy720000010.ogg"],
+		[
+			"惊慌的观众",
+			"「你这家伙不要靠近我,要是传染了可怎么办」",
+			"aiy730000010.ogg",
+		],
+		["粗鲁的观众", "「你这小鬼赶快滚开」", "aiy740000010.ogg"],
+		["凯伊姆", "「接下来」", "aiy310000360.ogg"], //男主36
+		["圣女", "「发生什么事了?看上去似乎很嘈杂」", "aiy030000030.ogg"], //圣女03
+		[
+			"随从",
+			"「似乎是某个人逃跑了······具体的我也不是很清楚」",
+			"aiy130000010.ogg",
+		], //侍从01
+		[
+			"神官",
+			"「圣女大人,继续待在天台上可能会出事,请您先回到室内吧」",
+			"aiy440000070.ogg",
+		], //神官07
+		[
+			"圣女",
+			"「不用在意我,比起那个,我更关心究竟发生了什么事」",
+			"aiy030000050.ogg",
+		], //圣女05
+		["神官", "「对不起,我真的不知道」", "aiy440000080.ogg"], //神官08
+		["圣女", "「······是吗」", "aiy030000060.ogg"], //圣女06
+		["男", "「恕我僭越,请准许我说明情况」", "aiy320000010.ogg"], //男主他哥01
+		["男", "「在来觐见的人群中出现了《羽化病》的患者」", "aiy320000020.ogg"], //男主他哥02
+		["男", "「围观的人群因而产生了骚动」", "aiy320000030.ogg"], //男主他哥03
+		[
+			"男",
+			"「现在,《防疫局》已经派遣了部队。我想不久之后,他们就会安静下来了」",
+			"aiy320000040.ogg",
+		], //男主他哥04
+		["圣女", "「羽化病······」", "aiy030000070.ogg"], //圣女07
+		["男", "「怎么了?」", "aiy320000050.ogg"], //男主他哥05
+		["圣女", "「没什么」", "aiy030000080.ogg"], //圣女08
+		["圣女", "「辛苦了,你的名字是?」", "aiy030000090.ogg"], //圣女09
+		[
+			"男",
+			"「属下是在防疫局任职的,鲁基乌斯· 迪斯·米利尤」",
+			"aiy320000060.ogg",
+		], //男主他哥06
+		[
+			"神官",
+			"「噢噢,阁下就是鲁基乌斯卿吗,我听说过你的传闻」",
+			"aiy440000090.ogg",
+		], //神官09
+		["神官", "「阁下是在工作上相当出色的人呢」", "aiy440000100.ogg"], //神官10
+		["鲁基乌斯", "「不敢当」", "aiy320000070.ogg"], //男主他哥07
+		[
+			"鲁基乌斯",
+			"「话说回来,这次是属下警备工作的失职。让圣女大人见到这不成体统的一面,请您见谅」",
+			"aiy320000080.ogg",
+		], //男主他哥08
+		[
+			"圣女",
+			"「即使是目不见物的我,也能感受到聚集于此的民众数量之多。警备工作难以展开也在情理之中」",
+			"aiy030000100.ogg",
+		], //圣女10
+		["鲁基乌斯", "「属下不胜惶恐」", "aiy320000090.ogg"], //男主他哥09
+		[
+			"鲁基乌斯",
+			"「接下来属下还要回到工作岗位上,在这里就先告退了」",
+			"aiy320000100.ogg",
+		], //男主他哥10
+		["圣女", "「鲁基乌斯先生」", "aiy030000110.ogg"], //圣女11
+		["鲁基乌斯", "「属下在」", "aiy320000110.ogg"], //男主他哥11
+		["圣女", "「你是怎样看待羽狩的工作的呢?」", "aiy030000120.ogg"], //圣女12
+		["神官", "「圣,圣女大人」", "aiy440000110.ogg"], //神官11
+		[
+			"鲁基乌斯",
+			"「防疫局的工作是国王陛下赐予的重要职务。属下非常荣幸能够为这个都市的繁荣尽一份微薄之力」",
+			"aiy320000120.ogg",
+		], //男主他哥12
+		["神官", "「不,不亏是鲁基乌斯卿,相当优秀的想法」", "aiy440000120.ogg"], //神官12
+		["圣女", "「是吗。辛苦你了」", "aiy030000130.ogg"], //圣女13
+		["随从", "「圣女大人······」", "aiy130000020.ogg"], //侍从02
+		["鲁基乌斯", "「······」", "aiy320000135.ogg"], //男主他哥13
+		["鲁基乌斯", "「那么,属下就回岗位去了」", "aiy320000140.ogg"], //男主他哥14
+	];
+	this.chapter03 = [
+		["", "从羽化病的少年纷乱的足音中,可以听得出相当的疲劳。"],
+		["", "显然,他并没有想到我会捷足先登吧。"],
+		["", "少年惶恐地回头看了一眼后,微微露出安心的表情,双手拄在膝盖上。。"],
+		["凯伊姆", "「辛苦你了」", "aiy310000370.ogg"], //男主37
+		["羽化病患少年", "「稀!?」", "aiy750000010.ogg"],
+		["凯伊姆", "「逃到贫民区是个不错的想法」", "aiy310000380.ogg"], //男主38
+		["羽化病患少年", "「你,你是,羽狩吗?」", "aiy750000020.ogg"],
+		["凯伊姆", "「不是」", "aiy310000390.ogg"], //男主39
+		[
+			"羽化病患少年",
+			"「什,什么啊······混蛋,不要吓我啊」",
+			"aiy750000030.ogg",
+		],
+		["凯伊姆", "「我对令你受惊这件事致以歉意」", "aiy310000400.ogg"], //男主40
+		["凯伊姆", "「作为回报,麻烦你把从店里偷的钱交出来吧」", "aiy310000410.ogg"], //男主41
+		["羽化病患少年", "「钱?你在说什么」", "aiy750000040.ogg"],
+		["凯伊姆", "「你要找的腰上的东西,掉在你身后了」", "aiy310000420.ogg"], //男主42
+		["羽化病患少年", "「哎?」", "aiy750000050.ogg"],
+		["羽化病患少年", "「呃呀」", "aiy750000060.ogg"],
+		["羽化病患少年", "「你······你这混蛋」", "aiy750000070.ogg"],
+		["凯伊姆", "「······」", "aiy310000430.ogg"], //男主43
+		["凯伊姆", "「把偷的钱交出来」", "aiy310000440.ogg"], //男主44
+		["羽化病患少年", "「我不知道你在······咕」", "aiy750000080.ogg"],
+		[
+			"羽化病患少年",
+			"「你,你说是我偷的······有什么证据吗」",
+			"aiy750000090.ogg",
+		],
+		["凯伊姆", "「你还挺倔的啊」", "aiy310000450.ogg"], //男主45
+		["凯伊姆", "「不过,给我听好了」", "aiy310000460.ogg"], //男主46
+		["凯伊姆", "「你偷的那些钱,是要上缴给《不蚀金锁》的上纳金」", "aiy310000470.ogg"], //男主47
+		["凯伊姆", "「而且,钱的主人是从前和吉克颇有渊源的女人」", "aiy310000480.ogg"], //男主48
+		["羽化病患少年", "「吉克?」", "aiy750000100.ogg"],
+		["凯伊姆", "「他是《不蚀金锁》的主人,这么说你就明白了吧」", "aiy310000490.ogg"], //男主49
+		["羽化病患少年", "「哎?哎?怎么会······」", "aiy750000110.ogg"],
+		["凯伊姆", "「再问你一遍,钱在哪里?」", "aiy310000510.ogg"], //男主51
+		["羽化病患少年", "「是,是,是,在我的怀里」", "aiy750000120.ogg"],
+		["凯伊姆", "「你没有擅自拿掉一部分吧」", "aiy310000520.ogg"], //男主52
+		["羽化病患少年", "「是,是的」", "aiy7500000130.ogg"],
+		[
+			"羽化病患少年",
+			"「那,那个,您是《不蚀金锁》的人吗?」",
+			"aiy750000140.ogg",
+		],
+		["凯伊姆", "「算是吧」", "aiy310000530.ogg"], //男主53
+		[
+			"羽化病患少年",
+			"「我什么都可以做,请您一定要帮帮我」",
+			"aiy750000150.ogg",
+		],
+		["凯伊姆", "「抱歉,我并没有兴趣去帮助他人」", "aiy310000540.ogg"], //男主54
+		[
+			"羽化病患少年",
+			"「我什么······什么,都会做的······」",
+			"aiy750000160.ogg",
+		],
+		["羽化病患少年", "「我一直都是生活在下层的」", "aiy750000170.ogg"],
+		[
+			"羽化病患少年",
+			"「可是,不知何时染上了羽化病······背后长出了翅膀······」",
+			"aiy750000180.ogg",
+		],
+		[
+			"羽化病患少年",
+			"「被寄宿工作的店赶了出来,只得流落到牢狱这里」",
+			"aiy750000190.ogg",
+		],
+		[
+			"羽化病患少年",
+			"「因为独自实在是饿的不行了,所以才会偷这些钱的」",
+			"aiy750000200.ogg",
+		],
+		[
+			"羽化病患少年",
+			"「我明明没有做任何坏事······为什么······会遇到这种事······」",
+			"aiy750000210.ogg",
+		],
+		["凯伊姆", "「谁知道」", "aiy310000550.ogg"], //男主55
+		[
+			"羽化病患少年",
+			"「呜······呜呜······接下来,要对我做什么?」",
+			"aiy750000220.ogg",
+		],
+		["凯伊姆", "「我要把你带到组织那里」", "aiy310000560.ogg"], //男主56
+		["羽化病患少年", "「怎,怎么这样」", "aiy750000230.ogg"],
+		["凯伊姆", "「不过,那样做的前提是你不是羽化病人」", "aiy310000570.ogg"], //男主57
+		["凯伊姆", "「组织也没有笨到把羽化病人招待到家里的程度」", "aiy310000580.ogg"], //男主58
+		["羽化病患少年", "「那么,是要放我逃走吗?」", "aiy750000240.ogg"],
+		["凯伊姆", "「我要让你学到教训」", "aiy310000590.ogg"], //男主59
+		["凯伊姆", "「如果换做是组织的制裁,至少要有断条胳膊的觉悟」", "aiy310000600.ogg"], //男主60
+		["凯伊姆", "「你的运气不错」", "aiy310000610.ogg"], //男主61
+		["羽化病患少年", "「唔······啊,是的······」", "aiy750000250.ogg"],
+		["凯伊姆", "「滚」", "aiy310000620.ogg"], //男主62
+		["羽化病患少年", "「非常感谢」", "aiy750000260.ogg"],
+		["羽化病患少年", "「唔啊!?」", "aiy750000270.ogg"],
+		["男", "「到这里就结束了,羽化病人」", "aiy430000010.ogg"], //兰格01
+		["男", "「确认他的翅膀」", "aiy430000020.ogg"], //兰格02
+		["", "趁还没有被卷入麻烦的事情之前,赶快离开这里吧"],
+		["羽狩的指挥者", "「那边的那个人」", "aiy430000030.ogg"], //兰格03
+		["凯伊姆", "「······有什么事?」", "aiy310000630.ogg"], //男主63
+		["羽狩的指挥者", "「可以稍微让我问几句话吗」", "aiy430000040.ogg"], //兰格04
+		["凯伊姆", "「······」", "aiy310000640.ogg"], //男主64
+		["凯伊姆", "「啊啊,无妨」", "aiy310000650.ogg"], //男主65
+		["羽狩的指挥者", "「感谢您的合作」", "aiy430000050.ogg"], //兰格05
+		["", "队长殷勤地致以谢礼。"],
+		["", "而在他的眼前,少年的衣服已经被他的补下们扯破。"],
+		["", "在瘦骨嶙峋的裸露后背上,长有纯白的羽翼。"],
+		["红发的羽狩", "「副队长,确认翅膀的持有了」"],
+		["羽狩的副队长", "「保护他」", "aiy430000060.ogg"], //兰格06
+		["羽化病患少年", "「不要······请原谅,我······」", "aiy750000280.ogg"],
+		[
+			"羽狩的副队长",
+			"「我们只是要带你去治愈院治疗羽化病,不是什么应该感到害怕的事情」",
+			"aiy430000070.ogg",
+		], //兰格07
+		["羽化病患少年", "「可是,可是」", "aiy750000290.ogg"],
+		["羽狩的副队长", "「没关系的」", "aiy430000080.ogg"], //兰格08
+		["羽化病患少年", "「······哥,哥哥」", "aiy750000300.ogg"],
+		["羽狩的副队长", "「你是羽化病人的亲属吗?」", "aiy430000090.ogg"], //兰格09
+		["凯伊姆", "「只是路人而已」", "aiy310000660.ogg"], //男主66
+		["凯伊姆", "「顺带一提,我也没有打算找你们的麻烦」", "aiy310000670.ogg"], //男主67
+		[
+			"羽狩的副队长",
+			"「前几天,有个和你说了同样的话的人,在我们背向他的瞬间对我们发动了袭击」",
+			"aiy430000100.ogg",
+		], //兰格10
+		["羽狩的副队长", "「我的一个部下就此永久失去了半截胳膊」", "aiy430000110.ogg"], //兰格11
+		["凯伊姆", "「我表示同情」", "aiy310000680.ogg"], //男主68
+		["凯伊姆", "「我马上就会消失的,这样就没问题了吧?」", "aiy310000690.ogg"], //男主69
+		["羽狩的副队长", "「嘛,不要这么慌张」", "aiy430000120.ogg"], //兰格12
+		["", "副队长看着羽化的少年。"],
+		["羽狩的副队长", "「你与这个人是什么关系?没有被他殴打吗?」", "aiy430000130.ogg"], //兰格13
+		["羽化病患少年", "「没,没有」", "aiy750000310.ogg"],
+		[
+			"羽狩的副队长",
+			"「如何对我们保持合作,你就可以在治愈院得到优先的治疗」",
+			"aiy430000140.ogg",
+		], //兰格14
+		["羽化病患少年", "「······」", "aiy750000320.ogg"],
+		[
+			"羽化病患少年",
+			"「那个人,是《不蚀金锁》的组织成员······」",
+			"aiy750000330.ogg",
+		],
+		[
+			"羽化病患少年",
+			"「突然说让我拿出钱来,我刚一拒绝他就打我」",
+			"aiy750000340.ogg",
+		],
+		["羽狩的副队长", "「原来如此······」", "aiy430000150.ogg"], //兰格15
+		[
+			"羽狩的副队长",
+			"「那位少年说你是《不蚀金锁》的一员,不知此事是否属实?」",
+			"aiy430000160.ogg",
+		], //兰格16
+		["凯伊姆", "「当然不是」", "aiy310000700.ogg"], //男主70
+		["凯伊姆", "「我只是从那里接受工作而已,并不是他们的成员」", "aiy310000710.ogg"], //男主71
+		["羽狩的副队长", "「你的意思是说,少年在说谎吗?」", "aiy430000170.ogg"], //兰格17
+		["凯伊姆", "「啊啊」", "aiy310000720.ogg"], //男主72
+		[
+			"凯伊姆",
+			"「如果你们和组织有关系的话,只要问问我是不是那里的成员就能明白事实了吧」",
+			"aiy310000730.ogg",
+		], //男主73
+		["羽狩的副队长", "「就算我去询问,也无法从他们那里得到事实」", "aiy430000180.ogg"], //兰格18
+		[
+			"羽狩的副队长",
+			"「《不蚀金锁》的那些人一向都不对我们合作,我对此深感困扰」",
+			"aiy430000190.ogg",
+		], //兰格19
+		["凯伊姆", "「真是辛苦啊」", "aiy310000740.ogg"], //男主74
+		["羽狩的副队长", "「说的是啊」", "aiy430000200.ogg"], //兰格20
+		["羽狩的副队长", "「其实,砍下我部下胳膊的似乎也是组织的成员呢」", "aiy430000210.ogg"], //兰格21
+		[
+			"羽狩的副队长",
+			"「无需如此警戒,我只是想在看守所向你咨询一些事情而已」",
+			"aiy430000220.ogg",
+		], //兰格22
+		[
+			"羽狩的副队长",
+			"「如果能够知晓牢狱与组织的事情,我们也可以尽可能地对更多的羽化病人进行保护」",
+			"aiy430000230.ogg",
+		], //兰格23
+		["羽狩的副队长", "「那和整条街道的和平也是紧密相关的吧?」", "aiy430000240.ogg"], //兰格24
+		["凯伊姆", "「我知道,你们有逮捕干扰狩猎羽化病人的权力」", "aiy310000750.ogg"], //男主75
+		[
+			"凯伊姆",
+			"「但是,我没有对你们做出任何干扰,为什么要对我如此纠缠不休呢」",
+			"aiy310000760.ogg",
+		], //男主76
+		["羽狩的副队长", "「那些话,我们会在看守所对你详细说明的」", "aiy430000250.ogg"], //兰格25
+		["凯伊姆", "「······」", "aiy310000770.ogg"], //男主77
+		["", "在这里起争执的话,就会被羽狩加害。"],
+		["", "就算能从这里脱身,今后只要碰面就会产生纠纷也是摆明的事情。"],
+		["", "就算逃跑,也没有好的结果。"],
+		["", "正当我想要打圆场的时候,刚才的气氛一瞬间产生了转变。"],
+		["", "发生了什么事······"],
+		["???", "「我认为,那位先生是正确的」", "aiy050000010.ogg"], //菲奥奈01
+		["", "羽狩们一起回头。"],
+		["", "而在他们视线的焦点处,"],
+		["", "伫立着一位女性。"],
+		["", "在端正的容颜下,代表着强烈意志的双眉十分显眼。"],
+		["", "身体的柔软与紧紧包裹在其身上的羽狩制服,两者显得十分的不搭配。"],
+		["", "我还是第一次看到女性的羽狩。"],
+		["羽狩的副队长", "「队长,这是获得《不蚀金锁》情报的好机会」", "aiy430000260.ogg"], //兰格26
+		[
+			"羽狩的队长",
+			"「兰格副队长,就算是为了获得情报,也不能做出恫吓的发言啊」",
+			"aiy050000020.ogg",
+		], //菲奥奈02
+		["兰格副队长", "「我并没有打算去恫吓他······」", "aiy430000270.ogg"], //兰格27
+		["羽狩的队长", "「告诉我那个被砍掉胳膊的队员的名字」", "aiy050000030.ogg"], //菲奥奈03
+		["羽狩的队长", "「我会去探望他的」", "aiy050000040.ogg"], //菲奥奈04
+		["兰格副队长", "「那个是······」", "aiy430000280.ogg"], //兰格28
+		["羽狩的队长", "「我知道,你一直在为有所收获而努力工作」", "aiy050000050.ogg"], //菲奥奈05
+		["羽狩的队长", "「但是,正因为我们的工作是为民众提供帮助」", "aiy050000060.ogg"], //菲奥奈06
+		["羽狩的队长", "「所以就更不能损害人与人之间的信赖」", "aiy050000070.ogg"], //菲奥奈07
+		["兰格副队长", "「我会铭记在心」", "aiy430000290.ogg"], //兰格29
+		["羽狩的队长", "「这位先生,我的部下失礼了」", "aiy050000080.ogg"], //菲奥奈08
+		["凯伊姆", "「只要不对我再来一次就好」", "aiy310000780.ogg"], //男主78
+		["羽狩的队长", "「请稍等」", "aiy050000090.ogg"], //菲奥奈09
+		["凯伊姆", "「有什么事?」", "aiy310000790.ogg"], //男主79
+		["羽狩的队长", "「我想确认一件事」", "aiy050000100.ogg"], //菲奥奈10
+		["羽狩的队长", "「你真的不是《不蚀金锁》的成员吗?」", "aiy050000110.ogg"], //菲奥奈11
+		["凯伊姆", "「真的」", "aiy310000800.ogg"], //男主80
+		["凯伊姆", "「如果我说是的话,你有什么打算?」", "aiy310000810.ogg"], //男主81
+		["羽狩的队长", "「我听过传闻,说他们是用依靠暴力而得的钱在生活」", "aiy050000120.ogg"], //菲奥奈12
+		["凯伊姆", "「······这样啊」", "aiy310000820.ogg"], //男主82
+		["凯伊姆", "「如果能有收获就好了啊」", "aiy310000830.ogg"], //男主83
+	];
+	this.chapter04 = [
+		["梅尔特", "「欢迎光临」", "aiy120000020.ogg"], //老板娘01,文件序号是以2开始,后续全部加1
+		["梅尔特", "「辛苦了」", "aiy120000030.ogg"], //老板娘02
+		["梅尔特", "「抱歉,又拜托给你了个这么麻烦的工作」", "aiy120000040.ogg"], //老板娘03
+		["凯伊姆", "「没什么,比想象中完成的更容易」", "aiy310000840.ogg"], //男主84
+		["梅尔特", "「那就好」", "aiy120000050.ogg"], //老板娘04
+		["梅尔特", "「这是我的一点谢意」", "aiy120000060.ogg"], //老板娘05
+		["凯伊姆", "「味道有些变化啊」", "aiy310000850.ogg"], //男主85
+		["梅尔特", "「啊,被发现了?」", "aiy120000070.ogg"], //老板娘06
+		["梅尔特", "「最近,没能到手什么好的原料呢」", "aiy120000080.ogg"], //老板娘07
+		["凯伊姆", "「去拜托吉克如何?」", "aiy310000860.ogg"], //男主86
+		[
+			"梅尔特",
+			"「话是这么说,但是总不能用店里采购的这种小事去麻烦他吧······」",
+			"aiy120000090.ogg"
+		], //老板娘08
+		["凯伊姆", "「那希望你也不要来麻烦我」", "aiy310000870.ogg"], //男主87
+		["梅尔特", "「那 是 两 码 事」", "aiy120000100.ogg"], //老板娘09
+		["梅尔特", "「再说,凯伊姆是靠着工作来生活的吧」", "aiy120000110.ogg"], //老板娘10
+		[
+			"梅尔特",
+			"「而且,自己的钱被偷了这么害羞的事,向凯伊姆以外的其他人都说不出口」",
+			"aiy120000120.ogg"
+		], //老板娘11
+		["凯伊姆", "「反正,也已经传到吉克的耳朵里了」", "aiy310000880.ogg"], //男主88
+		["梅尔特", "「这是面子问题啊,面子问题」", "aiy120000130.ogg"], //老板娘12
+		["凯伊姆", "「嘛,算了」", "aiy310000890.ogg"], //男主89
+		["凯伊姆", "「这样就好了吧?」", "aiy310000900.ogg"], //男主90
+		["梅尔特", "「这是钱包呢」", "aiy120000140.ogg"], //老板娘13
+		["梅尔特", "「嗯,东西没少」", "aiy120000150.ogg"], //老板娘14
+		["梅尔特", "「太好啦—这个月的上纳金,我可全部都放在里面了呢」", "aiy120000160.ogg"], //老板娘15
+		["梅尔特", "「如果没有找到的话,说不定就又会被送到娼馆里了呢」", "aiy120000170.ogg"], //老板娘16
+		["凯伊姆", "「在那边不是来钱更快吗?」", "aiy310000910.ogg"], //男主91
+		["梅尔特", "「阿拉,你是在说我还能有魅力吗?」", "aiy120000180.ogg"], //老板娘17
+		["凯伊姆", "「这是客套话而已」", "aiy310000920.ogg"], //男主92
+		["梅尔特", "「欺负人」", "aiy120000190.ogg"], //老板娘18
+		["梅尔特", "「总而言之,今天帮大忙了」", "aiy120000200.ogg"], //老板娘19
+		["梅尔特", "「谢礼嘛······」", "aiy120000210.ogg"], //老板娘20
+		["凯伊姆", "「就记在账单上吧」", "aiy310000930.ogg"], //男主93
+		["梅尔特", "「了解—盛谢惠顾了哦?」", "aiy120000220.ogg"], //老板娘21
+		["", "喀啷喀啷"],
+		["", "门铃响起"],
+		["", "喧哗瞬间安静下来。"],
+		["", "进来的人是吉克。"],
+		["", "是掌控着牢狱的组织之一,《不蚀金锁》的头目。"],
+		["", "不仅组织的成员,就连店内一般的客人也对他以注目礼表示敬意。"],
+		["吉克", "「各位继续吧」", "aiy340000010.ogg"], //吉克01
+		["", "仿佛停滞的时钟重新转动了一般,店内恢复了热闹的气氛。"],
+		["吉克", "「抱歉,今天拜托你去做了无聊的工作」。", "aiy340000020.ogg"], //吉克02
+		["凯伊姆", "「不用介意」", "aiy310000940.ogg"], //男主94
+		["", "吉克轻轻点了点头,在我右边坐了下来"],
+		["凯伊姆", "「逃跑的男人怎么样了?」", "aiy310000950.ogg"], //男主95
+		["吉克", "「嗯?已经不在这个世上了」。", "aiy340000030.ogg"], //吉克03
+		["吉克", "「有什么想要知道的事吗?」", "aiy340000040.ogg"], //吉克04
+		["凯伊姆", "「不,没什么」", "aiy310000960.ogg"], //男主96
+		["吉克", "「那个无聊的家伙,完全没有趣味呢」", "aiy340000050.ogg"], //吉克05
+		["吉克", "「真希望他也替我负责清扫的部下也考虑考虑」", "aiy340000060.ogg"], //吉克06
+		["凯伊姆", "「真是灾难啊」", "aiy310000970.ogg"], //男主97
+		["吉克", "「比起那个,我听说了哦。你去追羽化病人了啊」", "aiy340000070.ogg"], //吉克07
+		["凯伊姆", "「消息真灵通」", "aiy310000980.ogg"], //男主98
+		["吉克", "「梅尔特也注意点」", "aiy340000080.ogg"], //吉克08
+		["吉克", "「你丢钱已经不是一回两回了」", "aiy340000090.ogg"], //吉克09
+		["梅尔特", "「好的—我会注意的。」", "aiy120000230.ogg"], //老板娘22
+		["梅尔特", "「吉克还是平常的点单吧」", "aiy120000240.ogg"], //老板娘23
+		["梅尔特", "「凯伊姆要再来一杯吗?」", "aiy120000250.ogg"], //老板娘24
+		["", "我们用眼神点头示意后,梅尔特开始准备起酒来。"],
+		["", "悠然地吐出眼圈后,吉克取出一个纸包放在柜台上。"],
+		["吉克", "「这是抓捕逃跑男人的报酬」", "aiy340000100.ogg"], //吉克10
+		["凯伊姆", "「下次有什么事再告诉我」", "aiy310000990.ogg"], //男主99
+		["梅尔特", "「来,久等了」", "aiy120000260.ogg"], //老板娘25
+		["凯伊姆", "「话说回来梅尔特,为什么会被那种孩子偷到钱?」", "aiy310001000.ogg"], //男主100
+		["吉克", "「让我猜猜看」", "aiy340000110.ogg"], //吉克11
+		["吉克", "「是那个吧,看某个特立独行的男人入迷了,所以就有了空隙?」", "aiy340000120.ogg"], //吉克12
+		["梅尔特", "「可惜—」", "aiy120000270.ogg"], //老板娘26
+		["梅尔特", "「事实恰恰相反,是那家伙一直在纠缠我」", "aiy120000280.ogg"], //老板娘27
+		["凯伊姆", "「完全把你当成新进的女佣了么」", "aiy310001010.ogg"], //男主101
+		["梅尔特", "「我从前可是很有名的,不会被当成这种下人吧」", "aiy120000290.ogg"], //老板娘28
+		["梅尔特", "「······而且,我没法对对我这么钟情的人发火啊」", "aiy120000300.ogg"], //老板娘29
+		["凯伊姆&吉克", "「你傻啊」", "aiy310001027.ogg"], //男主102.7,吉克13.5
+		["梅尔特", "「异口同声呢,不亏是兄弟」", "aiy120000310.ogg"], //老板娘30
+		["凯伊姆", "「别用这种称呼,怪恶心的」", "aiy310001030.ogg"], //男主103
+		["吉克", "「说得没错」", "aiy340000140.ogg"], //吉克14
+		["吉克", "「······说起来······」", "aiy340000150.ogg"], //吉克15
+		["梅尔特", "「怎么了?」", "aiy120000320.ogg"], //老板娘31
+		["吉克", "「有件事我一直很在意,我和凯伊姆,哪个是哥哥啊?」", "aiy340000160.ogg"], //吉克16
+		["凯伊姆", "「你也说这么无聊的话题」", "aiy310001040.ogg"], //男主104
+		["吉克", "「不,这是很重要的事情」", "aiy340000170.ogg"], //吉克17
+		["吉克", "「梅尔特,事实是怎么样的?」", "aiy340000180.ogg"], //吉克18
+		["梅尔特", "「啊~是怎么样的呢~」", "aiy120000330.ogg"], //老板娘32
+		["梅尔特", "「我忘记了」", "aiy120000340.ogg"], //老板娘33
+		["吉克", "「骗人」", "aiy340000190.ogg"], //吉克19
+		["梅尔特", "「我说真的」", "aiy120000350.ogg"], //老板娘34
+		["梅尔特", "「嘛,如果想起来了的话,就算是卸载艺术上我也会公诸于众的」", "aiy120000360.ogg"], //老板娘35
+		["吉克", "「喔唷」", "aiy340000200.ogg"], //吉克20
+		["吉克", "「那么,我就不能比你先死了啊」", "aiy340000210.ogg"], //吉克21
+		["梅尔特", "「蒙你费心」", "aiy120000370.ogg"], //老板娘36
+		["梅尔特", "「顺带一提,有传言说吃过我们这里的料理后可以长生不老哦?」", "aiy120000380.ogg"], //老板娘37
+		["吉克", "「好,来两份炖菜,记得加腊肠」", "aiy340000220.ogg"], //吉克22
+		["梅尔特", "「多谢惠顾」", "aiy120000390.ogg"], //老板娘38
+		["", "微微一笑后,梅尔特去厨房传达点单。"],
+		["凯伊姆", "「吉克······」", "aiy310001050.ogg"], //男主105
+		["吉克", "「啊,不好了」", "aiy340000230.ogg"], //吉克23
+		["凯伊姆", "「话题扯远了」", "aiy310001060.ogg"], //男主106
+		["梅尔特", "「什么话题来着?」", "aiy120000400.ogg"], //老板娘39
+		["凯伊姆", "「关于为什么你的钱会被偷这件事」", "aiy310001070.ogg"], //男主107
+		["凯伊姆", "「丢钱的时候以你来说,应该不会全无察觉吧?」", "aiy310001080.ogg"], //男主108
+		["梅尔特", "「算是吧,被偷的时候确实也想过要抓住他」", "aiy120000410.ogg"], //老板娘40
+		["梅尔特", "「但是,我注意到了那个孩子背后的鼓起呢」", "aiy120000420.ogg"], //老板娘41
+		["凯伊姆", "「所以就不由自主地放他逃跑了?」", "aiy310001090.ogg"], //男主109
+		["吉克", "「就算你想羽化病人施恩,也不会得到任何回报哦」", "aiy340000240.ogg"], //吉克24
+		["梅尔特", "「我知道」", "aiy120000430.ogg"], //老板娘42
+		["梅尔特", "「正因为知道,所以之后才会拜托凯伊姆去将钱取回的」", "aiy120000440.ogg"], //老板娘43
+		["梅尔特", "「可是,呢······」", "aiy120000450.ogg"], //老板娘44
+		["梅尔特", "「果然还是很矛盾呢」", "aiy120000460.ogg"], //老板娘45
+		["梅尔特", "「明明是自己放他逃跑的,之后又拜托别人去抓他」", "aiy120000470.ogg"], //老板娘46
+		["梅尔特", "「但是,在那一刹那······应该说是,突然露出了真心吧」", "aiy120000480.ogg"], //老板娘47
+		["梅尔特", "「真的,只是自我满足而已」", "aiy120000490.ogg"], //老板娘48
+		["凯伊姆", "「对于那个孩子来说不是很幸运么」", "aiy310001100.ogg"], //男主110
+		["凯伊姆", "「在被《不蚀金锁》抓到之前,就被羽狩保护了」", "aiy310001110.ogg"], //男主111
+		["凯伊姆", "「现在应该已经躺在治愈院的床上了」", "aiy310001120.ogg"], //男主112
+		["吉克", "「如果被我们抓到的话,嘛,至少也会断掉一根胳膊吧」", "aiy340000250.ogg"], //吉克25
+		["凯伊姆", "「比起失去胳膊,这不是个很好地结局吗」", "aiy310001130.ogg"], //男主113
+		["吉克", "「多亏梅尔特的一念之善,那个小子的胳膊被救下来了」", "aiy340000260.ogg"], //吉克26
+		["吉克", "「对我来说,不能去管教他稍微有点可惜就是了」", "aiy340000270.ogg"], //吉克27
+		["凯伊姆", "「我姑且是给了他两三拳」", "aiy310001140.ogg"], //男主114
+		["吉克", "「你这不是很善解人意么」", "aiy340000280.ogg"], //吉克28
+		["梅尔特", "「没有帮他的忙啊」", "aiy120000500.ogg"], //老板娘49
+		["凯伊姆", "「你给我用常识考虑考虑」", "aiy310001150.ogg"], //男主115
+		["梅尔特", "「本来,就算羽化病人不被羽狩带走而导致羽化病扩散,在牢狱里死亡的理由也要多少就有多少」", "aiy120000510.ogg"], //老板娘50
+		["梅尔特", "「没事到如今多一个病人,也不会有什么改变的吧」", "aiy120000520.ogg"], //老板娘51
+		["梅尔特", "「那么,那么些羽化病人为什么就不能和我们一起开心地生活呢」", "aiy120000530.ogg"], //老板娘52
+		["吉克", "「像梅尔特这样思考的人少之又少」", "aiy340000290.ogg"], //吉克29
+		["凯伊姆", "「就算早晚都会死去,为了今日的苟活而努力拼搏也是人之常情」", "aiy310001160.ogg"], //男主116
+		["梅尔特", "「我知道」", "aiy120000540.ogg"], //老板娘53
+		["梅尔特", "「别介意,我刚才只是顺口说说」", "aiy120000550.ogg"], //老板娘54
+		["", "喀啷喀啷"],
+		["梅尔特", "「辛苦了,艾莉斯」", "aiy120000560.ogg"], //老板娘55
+		["艾莉斯", "「嗯」", "aiy020000190.ogg"], //医生19
+		["", "用不可爱的声音随口回应后,艾莉斯理所当然般地在我左边的座位坐下"],
+		["", "梅尔特什么都没有问,就开始准备起茶来"],
+		["梅尔特", "「多亏你去帮忙叫凯伊姆了呢」", "aiy120000570.ogg"], //老板娘56
+		["艾莉斯", "「不客气」", "aiy020000200.ogg"], //医生20
+		["艾莉斯", "「我从店里出去以后不久就看到他了,这是我们被命运之绳紧紧相连的证明呢」", "aiy020000210.ogg"], //医生21
+		["凯伊姆", "「那还真是糟糕」", "aiy310001170.ogg"], //男主117
+		["艾莉斯", "「用不着这么害羞的」", "aiy020000220.ogg"], //医生22
+		["凯伊姆", "「用茶堵上你的嘴吧」", "aiy310001180.ogg"], //男主118
+		["艾莉斯", "「好」", "aiy020000230.ogg"], //医生23
+		["梅尔特", "「司空见惯的风景呢」", "aiy120000580.ogg"], //老板娘57
+		["凯伊姆", "「你在那之后去做什么了?」", "aiy310001190.ogg"], //男主119
+		["艾莉斯", "「去莉莉乌姆照顾患病的女孩子了」", "aiy020000240.ogg"], //医生24
+		["艾莉斯", "「好像是毒品上瘾,费了很大劲才止住她的胡闹」", "aiy020000250.ogg"], //医生25
+		["艾莉斯", "「希望你能好好管理下自己店里的娼妇呢」", "aiy020000260.ogg"], //医生26
+		["吉克", "「抱歉啊」", "aiy340000300.ogg"], //吉克30
+		["吉克", "「为表歉意,就让我来给艾莉斯诊疗一下吧」", "aiy340000310.ogg"], //吉克31
+		["艾莉斯", "「去死吧」", "aiy020000270.ogg"], //医生27
+		["梅尔特", "「是什么药?」", "aiy120000590.ogg"], //老板娘58
+		["吉克", "「是最近上市的一种药」", "aiy340000320.ogg"], //吉克32
+		["吉克", "「虽然中毒的可能性很低,但只要中毒便会一发而不可收拾」", "aiy340000330.ogg"], //吉克33
+		["艾莉斯", "「知道了」", "aiy020000280.ogg"], //医生28
+		["吉克", "「虽然我也许没有时间去管教她,不过我还是会注意的」", "aiy340000340.ogg"], //吉克34
+		["艾莉斯", "「毒药的主人是吉克吗?」", "aiy020000290.ogg"], //医生29
+		["吉克", "「《不蚀金锁》不卖药,这是从先代定下的规矩」", "aiy340000350.ogg"], //吉克35
+		["艾莉斯", "「抱歉,开玩笑的」", "aiy020000300.ogg"], //医生30
+		["吉克", "「关于药的出处,这边也会调查的」", "aiy340000360.ogg"], //吉克36
+		["吉克", "「如果有什么传闻的话记得告诉我」", "aiy340000370.ogg"], //吉克37
+		["艾莉斯", "「知道了」", "aiy020000310.ogg"], //医生31
+		["艾莉斯", "「麻药中毒的人真麻烦呢,又不能完全治好」", "aiy020000320.ogg"], //医生32
+		["艾莉斯", "「既然要悄悄的打,那就也悄悄地去死不就好了」", "aiy020000330.ogg"], //医生33
+		["", "喀啷喀啷"],
+		["奥兹", "「吉克大人,抱歉打扰你开心的时光」", "aiy350000240.ogg"], //金锁高官24
+		["吉克", "「怎么了」", "aiy340000380.ogg"], //吉克38
+		["奥兹", "「请将耳朵凑过来」", "aiy350000250.ogg"], //金锁高官25
+		["吉克", "「······知道了」", "aiy340000390.ogg"], //吉克39
+		["吉克", "「凯伊姆,一会到娼馆来一趟」", "aiy340000400.ogg"], //吉克40
+		["凯伊姆", "「啊啊」", "aiy310001200.ogg"], //男主120
+		["吉克", "「噢,我都给忘了」", "aiy340000410.ogg"], //吉克41
+		["吉克", "「让这里的大家都开心一下」", "aiy340000420.ogg"], //吉克42
+		["梅尔特", "「收到」", "aiy120000600.ogg"], //老板娘59
+		["吉克", "「我走了」", "aiy340000430.ogg"], //吉克43
+		["奥兹", "「好」", "aiy350000260.ogg"], //金锁高官26
+		["", "梆梆"],
+		["梅尔特", "「大家静静,有个好消息」", "aiy120000610.ogg"], //老板娘60
+		["梅尔特", "「今天有个『大人物』要请客哦」", "aiy120000620.ogg"], //老板娘61
+		["艾莉斯", "「吉克还是一如既往的大方」", "aiy020000340.ogg"], //医生34
+		["梅尔特", "「果然,头目就应该像这样呢」", "aiy120000630.ogg"], //老板娘62
+		["凯伊姆", "「他的做法是继承先代的」", "aiy310001210.ogg"], //男主121
+		["梅尔特", "「又来了又来了」", "aiy120000640.ogg"], //老板娘63
+		["梅尔特", "「他想要赶上先代还早了十年呢」", "aiy120000650.ogg"], //老板娘64
+		["凯伊姆", "「有些太严厉了吧」", "aiy310001220.ogg"], //男主122
+		["艾莉斯", "「说起来,吉克有什么事」", "aiy020000350.ogg"], //医生35
+		["梅尔特", "「既然要叫上凯伊姆,再怎么说也不是什么小事吧?」", "aiy120000670.ogg"], //老板娘66
+		["凯伊姆", "「说的也是」", "aiy310001230.ogg"], //男主123
+		["艾莉斯", "「我期待看到你光荣负伤的样子」", "aiy020000360.ogg"], //医生36
+		["凯伊姆", "「不要为人的不幸祈求啊」", "aiy310001240.ogg"], //男主124
+		["梅尔特", "「来,喝完这杯就去加油工作吧」", "aiy120000680.ogg"], //老板娘67
+		["凯伊姆", "「啊啊」", "aiy310001250.ogg"], //男主125
+		["凯伊姆", "「我走了」", "aiy310001260.ogg"], //男主126
+		["艾莉斯", "「加油(受伤)吧」", "aiy020000370.ogg"], //医生37
+		["梅尔特", "「一路走好」", "aiy120000690.ogg"], //老板娘68
+	];
+	this.chapter05 = [
+		["", "今晚的娼馆街比平常来得更加热闹"],
+		["", "视线中可以看到很多生客。"],
+		["莉莎", "「呀——!」", "aiy150000010.ogg"], //三小只C01
+		["莉莎", "「呐,凯伊姆凯伊姆凯伊姆」", "aiy150000020.ogg"], //三小只C02
+		["莉莎", "「我难道没有魅力吗?」", "aiy150000030.ogg"], //三小只C03
+		["凯伊姆", "「突然之间怎么了」", "aiy310001270.ogg"], //男主127
+		["莉莎", "「因为因为因为,明明这么热闹,却只有我没有接到客人」", "aiy150000040.ogg"], //三小只C04
+		["莉莎", "「亏我还对他们说,我会用超绝的技巧让他们舒服得如同化作夜明的流星般呢」", "aiy150000050.ogg"], //三小只C05
+		["凯伊姆", "「那个台词太糟糕了」", "aiy310001280.ogg"], //男主128
+		["莉莎", "「哎—什么啊,这可是我昨天花了一晚上考虑出来的呢」", "aiy150000060.ogg"], //三小只C06
+		["凯伊姆", "「去睡觉,不要用你那空空如也的脑袋去考虑这种无聊事」", "aiy310001290.ogg"], //男主129
+		["莉莎", "「好过分!?」", "aiy150000070.ogg"], //三小只C07
+		["凯伊姆", "「你看看周围,今天有很多新客人吧?」", "aiy310001300.ogg"], //男主130
+		["莉莎", "「嗯~~」", "aiy150000080.ogg"], //三小只C08
+		["莉莎", "「啊,你这么一说还真是」", "aiy150000090.ogg"], //三小只C09
+		["凯伊姆", "「他们心里都很紧张,不可能会被你这种强拉的手法钓上钩吧」", "aiy310001310.ogg"], //男主131
+		["莉莎", "「这样啊,是不想欲仙欲死吗?」", "aiy150000100.ogg"], //三小只C10
+		["凯伊姆", "「不,没有男人不会希望那样的」", "aiy310001320.ogg"], //男主132
+		["凯伊姆", "「不过,强来是不行的。去让他们被温柔地包容着化作流星吧」", "aiy310001330.ogg"], //男主133
+		["莉莎", "「原—来如此」", "aiy150000110.ogg"], //三小只C11
+		["莉莎", "「凯伊姆肯定很有娼妇的才能哦」", "aiy150000120.ogg"], //三小只C12
+		["凯伊姆", "「没有」", "aiy310001340.ogg"], //男主134
+		["凯伊姆", "「好了,你赶紧去赚钱吧」", "aiy310001350.ogg"], //男主135
+		["莉莎", "「收到—」", "aiy150000130.ogg"], //三小只C13
+		["", "莉莎精神满满地回答后,就啪嗒啪嗒地跑开了。"],
+		["莉莎", "「呐—那边的大哥哥,和我一起化作星光吧—」", "aiy150000140.ogg"], //三小只C14
+		["", "这家伙"],
+		["", "非要用这么麻烦的说法么。"],
+		["", "咚"],
+		["莉莎", "「呀」", "aiy150000150.ogg"], //三小只C15
+		["大胡子醉汉", "「好疼」", "aiy800000010.ogg"],
+		["", "突然,莉莎与一个男人相撞。"],
+		["", "虽然男人可以摆出派头,但从那虚浮的脚步可以看出他相当的醉意。"],
+		["莉莎", "「好疼疼,对不起」", "aiy150000160.ogg"], //三小只C16
+		["大胡子醉汉", "「这不是道歉就能了事的事吧,大姐—」", "aiy800000020.ogg"],
+		["大胡子醉汉", "「哟,穿得相当清凉啊,喂」", "aiy800000030.ogg"],
+		["大肚子的醉汉", "「喂—怎么了?」", "aiy810000010.ogg"],
+		["大胡子醉汉", "「这位小姐特意往我身上撞呢」", "aiy800000040.ogg"],
+		["大肚子的醉汉", "「喂喂喂喂,你是要做什么」", "aiy810000020.ogg"],
+		["莉莎", "「呜哎—所以我都说对不起了啊」", "aiy150000170.ogg"], //三小只C17
+		["莉莎", "「对,对了。作为冲撞的补偿,我会给您提供很好的服务的,怎么样?」", "aiy150000180.ogg"], //三小只C18
+		["大胡子醉汉", "「是免费的吧?」", "aiy800000050.ogg"],
+		["莉莎", "「那个······这,这姑且也是工作······免费就太」", "aiy150000190.ogg"], //三小只C19
+		["大肚子的醉汉", "「真是的,娼妇就是这种人。钱,钱,钱,完全没有做人的诚意」", "aiy810000030.ogg"],
+		["大胡子醉汉", "「好—你这家伙。我给你钱,你现在就在这里给我服务」", "aiy800000060.ogg"],
+		["莉莎", "「这,这该怎么说呢,稍微有些过了吧,在这里做很害羞的」", "aiy150000200.ogg"], //三小只C20
+		["大胡子醉汉", "「你除了晃腰以外还有用得着脑子的地方吗?别说这种像个人说的话啊」", "aiy800000070.ogg"],
+		["大肚子的醉汉", "「啊啊?这些钱不够吗?」", "aiy810000040.ogg"],
+		["大胡子醉汉", "「来,赶快把脚张开让我O,你们最喜欢这种事了吧?」", "aiy800000080.ogg"],
+		["莉莎", "「是,是认真的吗······这些人」", "aiy150000210.ogg"], //三小只C21
+		["大肚子的醉汉", "「赶快做」", "aiy810000050.ogg"],
+		["莉莎", "「呃······」", "aiy150000220.ogg"], //三小只C22
+		["凯伊姆", "「喂」", "aiy310001360.ogg"], //男主136
+		["大肚子的醉汉", "「啊?」", "aiy810000060.ogg"],
+		["凯伊姆", "「即便是娼妇,也有不能出售的东西」", "aiy310001370.ogg"], //男主137
+		["", "喀"],
+		["", "我一拳击中男人被酒精染红的脸,"],
+		["大胡子醉汉", "「咦」", "aiy800000090.ogg"],
+		["", "嘎"],
+		["", "再一拳击中另一个人的腹部。"],
+		["大肚子的醉汉", "「咕······」", "aiy810000070.ogg"],
+		["大胡子醉汉", "「唔······」", "aiy800000100.ogg"],
+		["", "两人像两只毛虫般躺在地上蠕动着挣扎"],
+		["", "一时半会应该起不来了吧"],
+		["莉莎", "「凯伊姆先生,太迟了太迟了太迟了」", "aiy150000230.ogg"], //三小只C23
+		["凯伊姆", "「都是你闯出的祸吧」", "aiy310001380.ogg"], //男主138
+		["莉莎", "「话是,那么说,可是」", "aiy150000240.ogg"], //三小只C24
+		["凯伊姆", "「赶快回莉莉乌姆去」", "aiy310001390.ogg"], //男主139
+		["莉莎", "「唔,嗯。谢了,凯伊姆。再见」", "aiy150000250.ogg"], //三小只C25
+		["", "久留无用"],
+	];
+	this.chapter06 = [
+		["", "漂浮的紫烟与甜美香醇的空气。"],
+
+
+	];
+},
+    "动画及周期装备映射": function () {
+	// 在此增加新插件
+	this.equip = {
+		//所有回合中的装备及速度
+		sword1: { id: "jianji", speed: 10 },
+	};
+	this.heroanimate = {
+		//勇士武器对应的动画(key为主手武器ID,value为帧动画名)
+		sword1: "jianji2",
+	};
+	this.enemyanimate = {
+		//怪物对应的动画(key为怪物ID,value为帧动画名)
+		bat: "jianji2",
+	};
+	this.equipanimate = {
+		//勇士周期性装备对应的动画(key为装备Id,value为帧动画名)
+		shield0: "jianji2",
+	};
+},
+    "勇士法抗乘算叠加": function () {
+    // 在此增加新插件
+    items.prototype.compareEquipment = function (
+      compareEquipId,
+      beComparedEquipId
     ) {
-      const data = {
-        px: px,
-        py: py,
-        width: width,
-        height: height,
-        allFarme: allFarme,
-        imageList: imageList,
-        soundList: soundList,
-      };
-      core.setFlag("animate_" + name, data);
-    };
-    this.deleteanimate = function (name) {
-      core.setFlag("animate_" + name);
-    };
-    let thistime = 0;
-    this.playanimate = function (name, x, y, hero, scalex, scaley) {
-      const data = {
-        name: name,
-        x: x,
-        y: y,
-        hero: hero,
-        scalex: scalex,
-        scaley: scaley,
-        farme: 0,
-      };
+      var result = { value: {}, percentage: {} };
+      var first = core.material.items[compareEquipId],
+        second = core.material.items[beComparedEquipId];
+      for (var one in result) {
+        for (var name in core.status.hero) {
+          if (name === "mdef" && one === "percentage") {
+            var ans = 1;
+            if (first?.equip?.[one]?.[name])
+              ans *= 1 - (first.equip[one][name] || 0) / 100;
 
-      core.plugin.playing.add(data);
-    };
-    core.registerAnimationFrame("animateonmap", true, function (timestamp) {
-      if (timestamp - thistime > 1000 / 60) {
-        thistime = timestamp;
-        core.clearMap(anctx);
-        core.plugin.playing.forEach((one) => {
-          const data = flags["animate_" + one.name];
-          if (!data) {
-            core.plugin.playing.delete(one);
+            if (second?.equip?.[one]?.[name])
+              ans /= 1 - (second.equip[one][name] || 0) / 100;
+
+            if (ans != 1) result[one][name] = ans;
+          } else if (name === "mdef" && one === "value") {
+            var ans = 0;
+            if (first) ans -= ((first.equip || {})[one] || {})[name] || 0;
+            if (second) ans += ((second.equip || {})[one] || {})[name] || 0;
+            if (ans != 0) result[one][name] = ans;
           } else {
-            data.imageList.forEach(function (image) {
-              if (
-                one.farme >= (image.beforefarme ?? 0) &&
-                one.farme <= (image.afterfarme ?? data.allFarme)
-              ) {
-                const img = core.material.images.images?.[image.image];
-                if (img) {
-                  const gla = image.globalAlpha ?? 100;
-                  const agla = image.aglobalAlpha ?? gla,
-                    beforefarme = image.beforefarme ?? 0;
-                  const afterfarme = image.afterfarme ?? data.allFarme;
-
-                  anctx.globalAlpha =
-                    (gla +
-                      ((agla - gla) * (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1)) /
-                    100;
-
-                  const cx =
-                      (image.cx ?? 0) +
-                      (((image.acx ?? 0) - (image.cx ?? 0)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    cy =
-                      (image.cy ?? 0) +
-                      (((image.acy ?? 0) - (image.cy ?? 0)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    cw =
-                      (image.cw ?? img.width) +
-                      (((image.acw ?? img.width) - (image.cw ?? img.width)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    ch =
-                      (image.ch ?? img.height) +
-                      (((image.acw ?? img.height) - (image.cw ?? img.height)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    x =
-                      (image.x ?? 0) +
-                      (((image.ax ?? 0) - (image.x ?? 0)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    y =
-                      (image.y ?? 0) +
-                      (((image.ay ?? 0) - (image.y ?? 0)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    w =
-                      (image.w ?? one.width) +
-                      (((image.aw ?? one.width) - (image.w ?? one.width)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    h =
-                      (image.h ?? one.height) +
-                      (((image.aw ?? one.height) - (image.w ?? one.height)) *
-                        (one.farme - beforefarme)) /
-                        (afterfarme - beforefarme || 1),
-                    angle =
-                      (Math.PI *
-                        ((image.image.angle ?? 0) +
-                          (((image.aangle ?? 0) - (image.image.angle ?? 0)) *
-                            (one.farme - beforefarme)) /
-                            (afterfarme - beforefarme || 1))) /
-                      180;
-
-                  if (one.hero) {
-                    let sx, sy;
-                    if (core.status.heroMoving < 0) {
-                      sx = 0;
-                      sy = 0;
-                    } else {
-                      sx =
-                        core.utils.scan[core.status.hero.loc.direction].x *
-                        4 *
-                        core.status.heroMoving;
-                      sy =
-                        core.utils.scan[core.status.hero.loc.direction].y *
-                        4 *
-                        core.status.heroMoving;
-                    }
-                    const herox = core.status.hero.loc.x * 32 + 16 + sx;
-                    const heroy = core.status.hero.loc.y * 32 + 16 + sy;
-                    core.drawImage(
-                      anctx,
-                      img,
-                      cx,
-                      cy,
-                      cw,
-                      ch,
-                      herox + (x - data.px) * one.scalex,
-                      heroy + (y - data.py) * one.scaley,
-                      w * one.scalex,
-                      h * one.scaley,
-                      angle
-                    );
-                  } else {
-                    core.drawImage(
-                      anctx,
-                      img,
-                      cx,
-                      cy,
-                      cw,
-                      ch,
-                      one.x + (x - data.px) * one.scalex,
-                      one.y + (y - data.py) * one.scaley,
-                      w * one.scalex,
-                      h * one.scaley,
-                      angle
-                    );
-                  }
-                }
-              }
-            });
-            data.soundList.forEach(function (sound) {
-              const lisen =
-                sound.sound &&
-                core.material.sounds[sound.sound] &&
-                core.musicStatus.soundStatus;
-              if (one.farme == sound.startfarme && lisen) {
-                if (sound.stopbefore) core.stopSound();
-                core.playSound(sound.sound);
-              }
-            });
-            one.farme++;
-            if (one.farme > data.allFarme) core.plugin.playing.delete(one);
+            if (typeof core.status.hero[name] == "number") {
+              var ans = 0;
+              if (first) ans += ((first.equip || {})[one] || {})[name] || 0;
+              if (second) ans -= ((second.equip || {})[one] || {})[name] || 0;
+              if (ans != 0) result[one][name] = ans;
+            }
           }
-        });
+        }
       }
-    });
-  }
+      return result;
+    };
+    let a = 1;
+    items.prototype._loadEquipEffect = function (equipId, unloadEquipId) {
+      // 比较能力值
+      var result = core.compareEquipment(equipId, unloadEquipId);
+
+      for (var name in result.percentage) {
+        if (name === "mdef") {
+          a *= result.percentage[name];
+          core.setBuff(name, 1 - a);
+        } else {
+          core.addBuff(name, result.percentage[name] / 100);
+        }
+      }
+      for (var name in result.value)
+        core.status.hero[name] += result.value[name];
+    };
+  },
+    "攻速临界": function () {
+	// 在此增加新插件
+	//临界表
+	core.ui._drawBookDetail_turnAndCriticals = function (enemy, floorId, texts) {
+		// 临界表
+		var criticals = core.enemys.nextCriticals(enemy.id, 8, enemy.x, enemy.y, floorId).map(function (v) {
+			return core.formatBigNumber(v[0]) + ":" + core.formatBigNumber(v[1]);
+		});
+		while (criticals[0] == '0:0') criticals.shift();
+		texts.push("\r[#FF6A6A]\\d攻击临界表:\\d\r[]" + JSON.stringify(criticals));
+		var criticals_spell = core.nextCriticals_spell(enemy.id, 8, enemy.x, enemy.y, floorId).map(function (v) {
+			return core.formatBigNumber(v[0]) + ":" + core.formatBigNumber(v[1]);
+		});
+		while (criticals_spell[0] == '0:0') criticals_spell.shift();
+		texts.push("\r[#FF6A6A]\\d法强临界表:\\d\r[]" + JSON.stringify(criticals_spell));
+		var criticals_speed = core.nextCriticals_speed(enemy.id, 8, enemy.x, enemy.y, floorId).map(function (v) {
+			return core.formatBigNumber(v[0]) + ":" + core.formatBigNumber(v[1]);
+		});
+		while (criticals_speed[0] == '0:0') criticals_speed.shift();
+		texts.push("\r[#FF6A6A]\\d速度临界表:\\d\r[]" + JSON.stringify(criticals_speed));
+
+	}
+	//攻速临界计算
+	core.nextCriticals_speed = function (enemy, number, x, y, floorId) {
+		if (typeof enemy == 'string') enemy = core.material.enemys[enemy];
+		number = number || 1;
+
+		var info = core.getDamageInfo(enemy, null, x, y, floorId);
+		if (info == null) { // 如果未破防...
+			return [
+				['?', '?']
+			];
+		}
+
+		if (typeof info == 'number') {
+			return [
+				[0, 0]
+			];
+		}
+		return core.enemys._nextSpeedCriticals_useBinarySearch(enemy, info, number, x, y, floorId);
+	}
+	enemys.prototype._nextSpeedCriticals_useBinarySearch = function (enemy, info, number, x, y, floorId) {
+		var mon_hp = info.mon_hp,
+			hero_speed = core.status.hero.speed,
+			mon_def = info.mon_def,
+			pre = info.damage;
+		var list = [];
+		var start_speed = hero_speed;
+
+		var calNext = function (currSpeed, maxSpeed) {
+			var start = Math.floor(currSpeed),
+				end = Math.floor(maxSpeed);
+			if (start > end) return null;
+
+			while (start < end) {
+				var mid = Math.floor((start + end) / 2);
+				if (mid - start > end - mid) mid--;
+				var nextInfo = core.enemys.getDamageInfo(enemy, { "speed": mid }, x, y, floorId);
+				if (nextInfo == null || (typeof nextInfo == 'number')) return null;
+				if (pre > nextInfo.damage) end = mid;
+				else start = mid + 1;
+			}
+			var nextInfo = core.enemys.getDamageInfo(enemy, { "speed": start }, x, y, floorId);
+			return nextInfo == null || (typeof nextInfo == 'number') || nextInfo.damage >= pre ? null : [start, nextInfo.damage];
+		}
+		var currSpeed = start_speed;
+		while (true) {
+			var next = calNext(currSpeed + 1, Number.MAX_SAFE_INTEGER, pre);
+			if (next == null) break;
+			currSpeed = next[0];
+			pre = next[1];
+			list.push([currSpeed - hero_speed, info.damage - pre]);
+			if (pre <= 0 && !core.flags.enableNegativeDamage) break;
+			if (list.length >= number) break;
+		}
+		if (list.length == 0) list.push([0, 0]);
+		return list;
+	}
+	//法强临界计算
+	core.nextCriticals_spell = function (enemy, number, x, y, floorId) {
+		if (typeof enemy == 'string') enemy = core.material.enemys[enemy];
+		number = number || 1;
+
+		var info = core.getDamageInfo(enemy, null, x, y, floorId);
+		if (info == null) { // 如果未破防...
+			return [
+				['?', '?']
+			];
+		}
+
+		if (typeof info == 'number') {
+			return [
+				[0, 0]
+			];
+		}
+		return core.enemys._nextSpellCriticals_useBinarySearch(enemy, info, number, x, y, floorId);
+	}
+	enemys.prototype._nextSpellCriticals_useBinarySearch = function (enemy, info, number, x, y, floorId) {
+		var mon_hp = info.mon_hp,
+			hero_spell = core.status.hero.spell,
+			mon_def = info.mon_def,
+			turn = info.mon_turn,
+			pre = info.damage;
+		var list = [];
+		var start_spell = hero_spell;
+
+		var calNext = function (currSpell, maxSpell) {
+			var start = Math.floor(currSpell),
+				end = Math.floor(maxSpell);
+			if (start > end) return null;
+
+			while (start < end) {
+				var mid = Math.floor((start + end) / 2);
+				if (mid - start > end - mid) mid--;
+				var nextInfo = core.enemys.getDamageInfo(enemy, { "spell": mid }, x, y, floorId);
+				if (nextInfo == null || (typeof nextInfo == 'number')) return null;
+				if (turn > nextInfo.mon_turn) end = mid;
+				else start = mid + 1;
+			}
+			var nextInfo = core.enemys.getDamageInfo(enemy, { "spell": start }, x, y, floorId);
+			return nextInfo == null || (typeof nextInfo == 'number') || nextInfo.mon_turn >= turn ? null : [start, nextInfo.damage];
+		}
+		var currSpell = start_spell;
+		while (true) {
+			var next = calNext(currSpell + 1, Number.MAX_SAFE_INTEGER, pre, turn);
+			if (next == null) break;
+			currSpell = next[0];
+			pre = next[1];
+			list.push([currSpell - hero_spell, info.damage - pre]);
+			if (pre <= 0 && !core.flags.enableNegativeDamage) break;
+			if (list.length >= number) break;
+		}
+		if (list.length == 0) list.push([0, 0]);
+		return list;
+	}
+},
+    "手册(临时)": function () {
+	// 在此增加新插件
+	ui.prototype._drawBook_drawContent = function (index, enemy, top, left) {
+		var width = core._PX_ - left; // 9 : 8 : 8 划分三列
+		this._drawBook_drawRow1(index, enemy, top, left, width, top + 20);
+		this._drawBook_drawRow4(index, enemy, top, left, width, top + 38);
+		this._drawBook_drawRow2(index, enemy, top, left, width, top + 56);
+		this._drawBook_drawRow3(index, enemy, top, left, width, top + 74);
+		this._drawBook_drawRow5(index, enemy, top, left, width, top + 90);
+	}
+
+	ui.prototype._drawBook_drawRow4 = function (index, enemy, top, left, width, position) {
+		// 绘制第一行
+		core.setTextAlign('ui', 'left');
+		var b13 = this._buildFont(13, true),
+			f13 = this._buildFont(13, false);
+		var col1 = left,
+			col2 = left + width * 9 / 25,
+			col3 = left + width * 17 / 25;
+		core.fillText('ui', core.getStatusLabel('speed'), col1, position, '#DDDDDD', f13);
+		core.fillText('ui', core.formatBigNumber(enemy.speed || 0), col1 + 30, position, null, b13);
+		core.fillText('ui', core.getStatusLabel('spell'), col2, position, null, f13);
+		core.fillText('ui', core.formatBigNumber(enemy.spell || 0), col2 + 30, position, null, b13);
+		core.fillText('ui', core.getStatusLabel('mdef'), col3, position, null, f13);
+		core.fillText('ui', core.formatBigNumber(enemy.mdef || 0) + "%", col3 + 30, position, null, b13);
+	}
+	ui.prototype._drawBook_drawRow5 = function (index, enemy, top, left, width, position) {
+		// 绘制第一行
+		core.setTextAlign('ui', 'left');
+		var b13 = this._buildFont(13, true),
+			f13 = this._buildFont(13, false);
+		var col1 = left,
+			col2 = left + width * 13 / 25;
+		core.fillText('ui', '速度临界', col1 - 120, position, '#DDDDDD', f13);
+		core.fillText('ui', `[${core.formatBigNumber(enemy.criticalSpeed?.[0] || 0)},${core.formatBigNumber(enemy.criticalSpeed?.[1] || 0)}]`, col1 - 60, position, null, b13);
+
+		core.fillText('ui', "勇士出手次数", col1, position, '#DDDDDD', f13);
+		core.fillText('ui', core.getDamageInfo(enemy, null) ? core.getDamageInfo(enemy, null).hero_turn : '???', col1 + 80, position, null, b13);
+		core.fillText('ui', "怪物出手次数", col2, position, null, f13);
+		core.fillText('ui', core.getDamageInfo(enemy, null) ? core.getDamageInfo(enemy, null).mon_turn : '???', col2 + 80, position, null, b13);
+
+	}
+	ui.prototype._drawBook_pageinfo = function () {
+		var per_page = 4;
+		var padding_top = 12; // 距离顶端像素
+		var per_height = (core._PY_ - 32 - padding_top) / per_page;
+		return { per_page: per_page, padding_top: padding_top, per_height: per_height };
+	}
+},
+    "新怪物手册": function () {
+	// 在此增加新插件
+	const book = document.createElement("canvas"); //创建怪物手册画布
+	book.style.position = "absolute";
+	book.style.zIndex = 400;
+	book.style.display = "none";
+	book.id = "book";
+	main.dom.gameGroup.insertAdjacentElement("afterend", book);
+	book.style.top = "50%";
+	book.style.left = "50%";
+	book.style.transform = "translate(-50%,-50%)";
+	const ctx = book.getContext("2d");
+	main.dom.book = book
+
+	book.onclick = function (e) { //画布点击监听
+		try {
+			e.preventDefault();
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+				py = Math.floor((e.clientY - top) / core.domStyle.scale);
+
+			let x, y;
+			if (core.domStyle.isVertical) { //对竖屏进行坐标转换
+				x = py * 3;
+				y = 1248 - px * 3;
+			} else {
+				x = px * 3;
+				y = py * 3;
+			}
+			bookclick(x, y)
+		} catch (ee) {
+			main.log(ee);
+		}
+	}
+
+	function bookclick(x, y) { //点击画布(x,y)触发的效果
+		const makeBox = ([x, y], [w, h]) => { //创建点击检测区域
+			return [
+				[x, y],
+				[x + w, y + h],
+			];
+		};
+		const pos = [x, y]; //创建点击坐标pos
+		const inRect = ([x, y], [ //检测点击坐标是否在检测区域中
+			[sx, sy],
+			[dx, dy]
+		]) => {
+			return sx <= x && x <= dx && sy <= y && y <= dy;
+		};
+		/**
+		 * 使用范例
+		 *const backbox = makeBox([15, 35], [210, 90]);
+		 *if(inRect(pos,backbox)){}
+		 */
+	}
+	this.drawBook = function (floorId = core.status.floorId) {
+		if (core.domStyle.isVertical) { //对竖屏进行绘制坐标变换,只需要考虑横屏绘制,竖屏将自适应转置
+			ctx.canvas.width = 1248;
+			ctx.canvas.height = 2028;
+			ctx.save(); //保存设置
+			ctx.translate(1248, 0); //重新定位右上角为基准
+			ctx.rotate(Math.PI / 2); //旋转90度
+		} else {
+			ctx.canvas.width = 2028;
+			ctx.canvas.height = 1248;
+		}
+		//在这里写绘制内容,只写横屏就行
+
+		ctx.restore(); //恢复变换前的坐标,否则将连续转置
+	}
+},
+    "存读档": function () {
+	// 在此增加新插件
+	const saveLoad = document.createElement("canvas"); //创建存读档画布
+	saveLoad.style.position = "absolute";
+	saveLoad.style.zIndex = 390;
+	saveLoad.style.display = "none";
+	saveLoad.id = "saveLoad";
+	main.dom.gameGroup.insertAdjacentElement("afterend", saveLoad);
+	saveLoad.style.top = "50%";
+	saveLoad.style.left = "50%";
+	saveLoad.style.transform = "translate(-50%,-50%)";
+	const ctx = saveLoad.getContext("2d");
+	main.dom.saveLoad = saveLoad
+	let cg = false
+	let choose = false
+	//////存读档界面更新
+	class SL {
+		constructor() {
+			this.index = 0;
+			this.refresh = false;
+			this.page = 0;
+			this.max_page = 1000
+			this.id = ""
+
+		}
+		update() {
+			core.clearMap(ctx)
+			core.ui._drawSLPanel_draw(this.page, this.max_page)
+		}
+		setIndex(index = this.index, refresh = this.refresh) {
+			this.index = index
+			core.saves.saveIndex = index
+			this.refresh = refresh
+		}
+		setPage(page = this.page) {
+			this.page = page
+		}
+		setOffset(offset = this.offset) {
+			this.offset = offset
+		}
+		setMax_page(max_page = this.max_page) {
+			this.max_page = max_page
+		}
+		setId(id) {
+			this.id = id
+		}
+	}
+	core.saveLoad = new SL()
+	saveLoad.onclick = function (e) { //画布点击监听
+		try {
+			e.preventDefault();
+			const left = core.dom.gameGroup.offsetLeft;
+			const top = core.dom.gameGroup.offsetTop;
+			const px = Math.floor((e.clientX - left) / core.domStyle.scale),
+				py = Math.floor((e.clientY - top) / core.domStyle.scale);
+
+			let x, y;
+			if (core.domStyle.isVertical) { //对竖屏进行坐标转换
+				x = py * 3;
+				y = 1248 - px * 3;
+			} else {
+				x = px * 3;
+				y = py * 3;
+			}
+			saveLoadclick(x, y)
+		} catch (ee) {
+			main.log(ee);
+		}
+	}
+	////// 删除某个存档
+	control.prototype.removeSave = function (index, callback) {
+		if (index == 0 || index == "autoSave") {
+			index = "autoSave";
+			core.removeLocalForage(index, function () {
+				core.saves.autosave.data = null;
+				core.saves.autosave.updated = false;
+				if (callback) callback();
+			});
+			return;
+		}
+		core.removeLocalForage(
+			"save" + index,
+			function () {
+				core.saves.favorite = core.saves.favorite.filter(function (i) {
+					return core.hasSave(i);
+				});
+				delete core.saves.favoriteName[index];
+				core.setLocalStorage("_slText" + index)
+				core.control._updateFavoriteSaves();
+				if (callback) callback();
+			},
+			function () {
+				core.playSound("操作失败");
+				core.drawTip("无法删除存档!");
+				if (callback) callback();
+			}
+		);
+	};
+	control.prototype._doSL_save = function (id) {
+		if (id == "autoSave") {
+			core.playSound("操作失败");
+			return core.drawTip("不能覆盖自动存档!");
+		}
+		// 在事件中的存档
+		if (core.status.event.interval != null)
+			core.setFlag("__events__", core.status.event.interval);
+		var data = core.saveData();
+		if (core.isReplaying() && core.status.replay.toReplay.length > 0) {
+			data.__toReplay__ = core.encodeRoute(core.status.replay.toReplay);
+		}
+		core.setLocalStorage("_slText" + id)
+		core.setLocalForage(
+			"save" + id,
+			data,
+			function () {
+				core.saves.saveIndex = id;
+				core.setLocalStorage("saveIndex", core.saves.saveIndex);
+				// 恢复事件
+				if (!core.events.recoverEvents(core.status.event.interval))
+					core.ui.closePanel();
+				core.playSound("存档");
+				core.drawTip("存档成功!");
+			},
+			function (err) {
+				console.error(err);
+				alert("存档失败,错误信息:\n" + err);
+			}
+		);
+		core.removeFlag("__events__");
+		return;
+	};
+
+	function saveLoadclick(x, y) { //点击画布(x,y)触发的效果
+		const makeBox = ([x, y], [w, h]) => { //创建点击检测区域
+			return [
+				[x, y],
+				[x + w, y + h],
+			];
+		};
+		const pos = [x, y]; //创建点击坐标pos
+		const inRect = ([x, y], [ //检测点击坐标是否在检测区域中
+			[sx, sy],
+			[dx, dy]
+		]) => {
+			return sx <= x && x <= dx && sy <= y && y <= dy;
+		};
+		let page = core.status.event.data.page,
+			offset = core.status.event.data.offset;
+		let index = page * 10 + offset;
+
+		const upbox = makeBox([675, 370], [100, 100]);
+		const upupbox = makeBox([675, 260], [100, 110]);
+		const upupupbox = makeBox([675, 150], [100, 120]);
+		const downbox = makeBox([675, 660], [100, 100]);
+		const downdownbox = makeBox([675, 760], [100, 110]);
+		const downdowndownbox = makeBox([675, 870], [100, 120]);
+		const addbox = makeBox([650, 1010], [150, 100]);
+		const modebox = makeBox([30, 124], [270, 1000]);
+		const save0box = makeBox([300, 100], [350, 124]);
+		const savebox = makeBox([300, 224], [350, 800]);
+		const save9box = makeBox([300, 1024], [350, 124]);
+		const backbox = makeBox([1750, 1150], [200, 100]);
+		const savenamebox = makeBox([800, 970], [250, 70]);
+		const classnamebox = makeBox([800, 1070], [250, 70]);
+		const savetextbox = makeBox([1050, 982], [640, 150]);
+		const cgbox = makeBox([1200, 1150], [300, 100])
+		const changebox = makeBox([1700, 970], [200, 70])
+		const deletebox = makeBox([1700, 1070], [200, 70])
+		const gobox = makeBox([800, 100], [1050, 870])
+		const favbox = makeBox([900, 300], [500, 110])
+		const class1box = makeBox([900, 410], [500, 110])
+		const class2box = makeBox([900, 520], [500, 110])
+		const class3box = makeBox([900, 630], [500, 110])
+		const class4box = makeBox([900, 740], [500, 110])
+		const class5box = makeBox([1400, 300], [500, 110])
+		const class6box = makeBox([1400, 410], [500, 110])
+		const class7box = makeBox([1400, 520], [500, 110])
+		const class8box = makeBox([1400, 630], [500, 110])
+		const changebackbox = makeBox([1700, 740], [130, 110])
+
+		if (inRect(pos, upbox)) {
+			choose = false
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page - 1) + offset);
+			return;
+		}
+		if (inRect(pos, upupbox)) {
+			choose = false
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page - 10) + offset);
+			return;
+		}
+		if (inRect(pos, upupupbox)) {
+			choose = false
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page - 100) + offset);
+			return;
+		}
+		if (inRect(pos, downbox)) {
+			choose = false
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page + 1) + offset);
+			return;
+		}
+		if (inRect(pos, downdownbox)) {
+			choose = false
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page + 10) + offset);
+			return;
+		}
+		if (inRect(pos, downdowndownbox)) {
+			choose = false
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page + 100) + offset);
+			return;
+		}
+		if (inRect(pos, backbox)) {
+			choose = false
+			if (core.events.recoverEvents(core.status.event.interval))
+				return;
+			core.ui.closePanel();
+			delete core.status.tempRoute;
+			if (!core.isPlaying())
+				core.showStartAnimate(true);
+			return;
+		}
+		if (inRect(pos, gobox) && !choose) {
+			if (offset == 0)
+				core.doSL("autoSave", core.status.event.id);
+			else {
+				var id = 9 * page + offset;
+				if (core.status.event.data.mode == 'fav') id = core.saves.favorite[id - 1];
+				if (core.status.event.data.mode !== 'fav' && core.status.event.data.mode !== 'all') id = core.saves[core.status.event.data.mode][id - 1];
+				core.doSL(id, core.status.event.id);
+			}
+			return;
+		}
+		if (inRect(pos, addbox) && core.status.event.data.mode === "all" && core.getLocalStorage("save_page", 0) !== 8000) {
+			core.playSound('确定');
+			core.setLocalStorage("save_page", core.getLocalStorage("save_page", 0) + 1000)
+			core.ui._drawSLPanel(10 * page + offset, true);
+			return;
+		}
+		if (inRect(pos, modebox)) {
+			choose = false
+			let mode = Math.floor((y - 124) / 100) - 1
+			if (mode < 0 && core.status.event.data.mode !== "all") {
+				core.status.event.data.mode = "all"
+				page = parseInt((core.saves.saveIndex - 1) / 9);
+				offset = core.saves.saveIndex - 9 * page;
+				core.ui._drawSLPanel(0, true);
+				return;
+			}
+			if (mode === 0 && core.status.event.data.mode !== "fav") {
+				core.status.event.data.mode = "fav"
+				core.ui._drawSLPanel(0, true);
+				return;
+			}
+			if (mode > 0 && core.status.event.data.mode !== `class_${mode}`) {
+				core.status.event.data.mode = `class_${mode}`
+				core.ui._drawSLPanel(0, true);
+				return;
+			}
+			return;
+		}
+
+		if (inRect(pos, save0box)) {
+			choose = false
+			core.saveLoad.setId("autoSave")
+			if (offset === 0) {
+				core.doSL("autoSave", core.status.event.id)
+				return
+			}
+			core.ui._drawSLPanel(10 * page)
+			return;
+		}
+		if (inRect(pos, savebox)) {
+			choose = false
+			let a = Math.ceil((y - 224) / 100)
+			let real_id = 9 * page + a
+			if (real_id == index) {
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[a - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][a - 1]
+				}
+				core.saveLoad.setId(real_id)
+				core.doSL(real_id, core.status.event.id)
+				return
+			} else {
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[a - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][a - 1]
+				}
+				core.saveLoad.setId(real_id)
+			}
+			core.ui._drawSLPanel(10 * page + a)
+			return;
+		}
+		if (inRect(pos, save9box)) {
+			choose = false
+			let real_id = 9 * page + 9
+			if (real_id == index) {
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + 8]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + 8]
+				}
+				core.saveLoad.setId(real_id)
+				core.doSL(real_id, core.status.event.id)
+				return
+			} else {
+				real_id = core.saves.favorite[9 * page + 8]
+
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + 8]
+				}
+				core.saveLoad.setId(real_id)
+
+			}
+			core.ui._drawSLPanel(10 * page + 9)
+			return;
+		}
+		if (inRect(pos, changebox)) {
+			choose = !choose
+			core.ui._drawSLPanel(index)
+			return;
+		}
+		if (inRect(pos, deletebox)) {
+			choose = false
+			let real_id = 9 * page + offset
+			if (core.status.event.data.mode === "fav") {
+				real_id = core.saves.favorite[9 * page + offset - 1]
+			}
+			if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+				real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+			}
+			core.removeSave(real_id, function () {
+				core.ui._drawSLPanel(index, true);
+			});
+			return;
+		}
+		if (inRect(pos, cgbox)) {
+			choose = false
+			cg = !cg
+			core.ui._drawSLPanel(10 * page + offset, true)
+			return;
+		}
+		if (inRect(pos, savenamebox) && offset !== 0) {
+			choose = false
+
+			let real_id = 9 * page + offset
+			if (core.status.event.data.mode === "fav") {
+				real_id = core.saves.favorite[9 * page + offset - 1]
+			}
+			if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+				real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+			}
+
+
+			core.myprompt("请输入想要显示的存档名(长度不超过5字符)", null, function (value) {
+				if (value && value.length <= 5) {
+					core.saves.favoriteName[real_id] = value;
+					core.control._updateFavoriteSaves();
+					core.ui._drawSLPanel(10 * page + offset);
+				} else if (value) {
+					alert("无效的输入!");
+				} else if (value === "") {
+					delete core.saves.favoriteName[real_id];
+					core.control._updateFavoriteSaves();
+					core.ui._drawSLPanel(10 * page + offset);
+				}
+			});
+
+			return;
+		}
+		if (inRect(pos, savetextbox) && offset !== 0) {
+			choose = false
+
+			let real_id = 9 * page + offset
+			if (core.status.event.data.mode === "fav") {
+				real_id = core.saves.favorite[9 * page + offset - 1]
+			}
+			if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+				real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+			}
+
+
+			core.myprompt("请输入存档笔记(长度不超过60字符)", null, function (value) {
+				if (value && value.length <= 60) {
+					core.setLocalStorage("_slText" + real_id, value)
+					core.ui._drawSLPanel(10 * page + offset);
+				} else if (value) {
+					alert("无效的输入!");
+				} else if (value === "") {
+					core.setLocalStorage("_slText" + real_id)
+					core.ui._drawSLPanel(10 * page + offset);
+				}
+			});
+
+			return;
+		}
+		if (inRect(pos, classnamebox) && core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+			choose = false
+
+			core.myprompt("请输入想要显示的存档分类(长度不超过5字符)", null, function (value) {
+				if (value && value.length <= 5) {
+					core.setLocalStorage("slClass_" + core.status.event.data.mode[6], value)
+					core.ui._drawSLPanel(10 * page + offset);
+				} else if (value) {
+					alert("无效的输入!");
+				} else if (value === "") {
+					core.setLocalStorage("slClass_" + core.status.event.data.mode[6])
+					core.ui._drawSLPanel(10 * page + offset);
+				}
+			});
+
+			return;
+		}
+		if (choose) {
+			if (inRect(pos, changebackbox)) {
+				choose = false
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, favbox)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+
+				let v = core.saves.favorite.indexOf(real_id)
+				if (v >= 0) { // 已经处于收藏状态:取消收藏
+					core.saves.favorite.splice(v, 1);
+					if (core.status.event.data.mode === "fav") choose = false
+				} else if (core.hasSave(real_id)) { // 存在存档则进行收藏
+
+					core.saves.favorite.push(real_id);
+
+					core.saves.favorite = core.saves.favorite.sort(function (a, b) { return a - b; }); // 保证有序
+
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class1box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_1.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_1.splice(v, 1);
+					if (core.status.event.data.mode === "class_1") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_1.push(real_id);
+					core.saves.class_1 = core.saves.class_1.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class2box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_2.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_2.splice(v, 1);
+					if (core.status.event.data.mode === "class_2") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_2.push(real_id);
+					core.saves.class_2 = core.saves.class_2.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class3box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_3.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_3.splice(v, 1);
+					if (core.status.event.data.mode === "class_3") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_3.push(real_id);
+					core.saves.class_3 = core.saves.class_3.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class4box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_4.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_4.splice(v, 1);
+					if (core.status.event.data.mode === "class_4") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_4.push(real_id);
+					core.saves.class_4 = core.saves.class_4.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class5box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_5.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_5.splice(v, 1);
+					if (core.status.event.data.mode === "class_5") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_5.push(real_id);
+					core.saves.class_5 = core.saves.class_5.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class6box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_6.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_6.splice(v, 1);
+					if (core.status.event.data.mode === "class_6") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_6.push(real_id);
+					core.saves.class_6 = core.saves.class_6.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class7box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_7.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_7.splice(v, 1);
+					if (core.status.event.data.mode === "class_7") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_7.push(real_id);
+					core.saves.class_7 = core.saves.class_7.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+			if (inRect(pos, class8box)) {
+				let real_id = 9 * page + offset
+				if (core.status.event.data.mode === "fav") {
+					real_id = core.saves.favorite[9 * page + offset - 1]
+				}
+				if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+					real_id = core.saves[core.status.event.data.mode][9 * page + offset - 1]
+				}
+				let v = core.saves.class_8.indexOf(real_id)
+				if (v >= 0) {
+					core.saves.class_8.splice(v, 1);
+					if (core.status.event.data.mode === "class_8") choose = false
+				} else if (core.hasSave(real_id)) {
+					core.saves.class_8.push(real_id);
+					core.saves.class_8 = core.saves.class_8.sort(function (a, b) { return a - b; }); // 保证有序
+				}
+				core.control._updateFavoriteSaves()
+				core.ui._drawSLPanel(index);
+				return
+			}
+		}
+	}
+	///// 读取收藏信息
+	control.prototype._loadFavoriteSaves = function () {
+		core.saves.favorite = core.getLocalStorage("favorite", []);
+		// --- 移除不存在的收藏
+		core.saves.favorite = core.saves.favorite.filter(function (i) {
+			return core.hasSave(i);
+		});
+		core.saves.favoriteName = core.getLocalStorage("favoriteName", {});
+	};
+
+
+	///// 读取分类信息
+	control.prototype._loadClassSaves = function () {
+		for (let i = 1; i < 9; i++) {
+			core.saves["class_" + i] = core.getLocalStorage("class_" + i, [])
+			core.saves["class_" + i] = core.saves["class_" + i].filter(function (i) {
+				return core.hasSave(i);
+			})
+		}
+
+	};
+
+	control.prototype._updateFavoriteSaves = function () {
+		core.setLocalStorage("favorite", core.saves.favorite);
+		for (let i = 1; i < 9; i++) {
+			core.setLocalStorage("class_" + i, core.saves["class_" + i])
+		}
+		core.setLocalStorage("favoriteName", core.saves.favoriteName);
+	};
+	////// 绘制存档/读档界面 //////
+	ui.prototype._drawSLPanel = function (index, refresh) {
+		core.control._updateFavoriteSaves()
+		core.control._loadClassSaves();
+		core.control._loadFavoriteSaves();
+
+		if (index == null) index = 1;
+		if (index < 0) index = 0;
+
+		var page = parseInt(index / 10),
+			offset = index % 10;
+		var max_page = (main.savePages + core.getLocalStorage("save_page", 0)) || 30;
+		if (core.status.event.data && core.status.event.data.mode == 'fav')
+			max_page = Math.ceil((core.saves.favorite || []).length / 9);
+		if (page >= max_page) page = max_page - 1;
+		if (core.status.event.data && core.status.event.data.mode == 'fav' && page == max_page - 1) {
+			offset = Math.min(offset, (core.saves.favorite || []).length - 9 * page);
+		}
+		if (core.status.event.data && (core.status.event.data.mode !== 'fav' && core.status.event.data.mode !== 'all'))
+			max_page = Math.ceil((core.saves[core.status.event.data.mode] || []).length / 9);
+		if (page >= max_page) page = max_page - 1;
+		if (core.status.event.data && (core.status.event.data.mode !== 'fav' && core.status.event.data.mode !== 'all') && page == max_page - 1) {
+			offset = Math.min(offset, (core.saves[core.status.event.data.mode] || []).length - 9 * page);
+		}
+
+		var last_page = -1;
+		var mode = 'all';
+		if (core.status.event.data) {
+			last_page = core.status.event.data.page;
+			mode = core.status.event.data.mode;
+		}
+		core.saveLoad.setIndex(index, refresh)
+		core.saveLoad.setPage(page)
+		core.saveLoad.setOffset(offset)
+		core.saveLoad.setMax_page(max_page)
+		core.status.event.data = { 'page': page, 'offset': offset, 'mode': mode };
+		core.status.event.ui = core.status.event.ui || [];
+		saveLoad.style.display = "block"
+		if (refresh || page != last_page) {
+			core.status.event.ui = [];
+			this._drawSLPanel_loadSave(page, function () {
+				core.ui._drawSLPanel_draw(page, max_page);
+			});
+		} else this._drawSLPanel_draw(page, max_page);
+	}
+
+	ui.prototype._drawSLPanel_draw = function (page, max_page) {
+		if (core.domStyle.isVertical) {
+			ctx.canvas.width = 1248;
+			ctx.canvas.height = 2028;
+			ctx.save(); //保存设置
+			ctx.translate(1248, 0); //重新定位右上角为基准
+			ctx.rotate(Math.PI / 2); //旋转90度
+		} else {
+			ctx.canvas.width = 2028;
+			ctx.canvas.height = 1248;
+		}
+		// --- 绘制背景
+		this._drawSLPanel_drawBackground();
+
+		// --- 绘制文字
+		core.setTextAlign(ctx, 'center');
+
+		var bottom = 1248 - 39;
+		core.fillBoldText1(
+			ctx,
+			"返回游戏",
+			1850,
+			bottom,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+
+		if (core.status.event.selection)
+			core.setFillStyle(ctx, '#FF6A6A');
+		ctx.strokeStyle = "#DDDDDD";
+		ctx.fillStyle = "#BBBB44"
+		ctx.lineWidth = 6
+		ctx.beginPath();
+
+		ctx.moveTo(300, 124)
+		ctx.lineTo(60, 124)
+		ctx.arcTo(30, 124, 30, 224, 30)
+		ctx.lineTo(30, 194)
+		ctx.arcTo(30, 224, 300, 224, 30)
+		ctx.lineTo(300, 224)
+		ctx.stroke()
+		if (core.status.event.data.mode === "all") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 224)
+		ctx.lineTo(60, 224)
+		ctx.arcTo(30, 224, 30, 324, 30)
+		ctx.lineTo(30, 294)
+		ctx.arcTo(30, 324, 300, 324, 30)
+		ctx.lineTo(300, 324)
+		ctx.stroke()
+		if (core.status.event.data.mode === "fav") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 324)
+		ctx.lineTo(60, 324)
+		ctx.arcTo(30, 324, 30, 424, 30)
+		ctx.lineTo(30, 394)
+		ctx.arcTo(30, 424, 300, 424, 30)
+		ctx.lineTo(300, 424)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_1") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 424)
+		ctx.lineTo(60, 424)
+		ctx.arcTo(30, 424, 30, 524, 30)
+		ctx.lineTo(30, 494)
+		ctx.arcTo(30, 524, 300, 524, 30)
+		ctx.lineTo(300, 524)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_2") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 524)
+		ctx.lineTo(60, 524)
+		ctx.arcTo(30, 524, 30, 624, 30)
+		ctx.lineTo(30, 594)
+		ctx.arcTo(30, 624, 300, 624, 30)
+		ctx.lineTo(300, 624)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_3") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 624)
+		ctx.lineTo(60, 624)
+		ctx.arcTo(30, 624, 30, 724, 30)
+		ctx.lineTo(30, 694)
+		ctx.arcTo(30, 724, 300, 724, 30)
+		ctx.lineTo(300, 724)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_4") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 724)
+		ctx.lineTo(60, 724)
+		ctx.arcTo(30, 724, 30, 824, 30)
+		ctx.lineTo(30, 794)
+		ctx.arcTo(30, 824, 300, 824, 30)
+		ctx.lineTo(300, 824)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_5") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 824)
+		ctx.lineTo(60, 824)
+		ctx.arcTo(30, 824, 30, 924, 30)
+		ctx.lineTo(30, 894)
+		ctx.arcTo(30, 924, 300, 924, 30)
+		ctx.lineTo(300, 924)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_6") ctx.fill()
+		ctx.beginPath();
+
+		ctx.moveTo(300, 924)
+		ctx.lineTo(60, 924)
+		ctx.arcTo(30, 924, 30, 1024, 30)
+		ctx.lineTo(30, 994)
+		ctx.arcTo(30, 1024, 300, 1024, 30)
+		ctx.lineTo(300, 1024)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_7") ctx.fill()
+		ctx.beginPath()
+		ctx.moveTo(300, 1024)
+		ctx.lineTo(60, 1024)
+		ctx.arcTo(30, 1024, 30, 1124, 30)
+		ctx.lineTo(30, 1094)
+		ctx.arcTo(30, 1124, 300, 1124, 30)
+		ctx.lineTo(300, 1124)
+		ctx.stroke()
+		if (core.status.event.data.mode === "class_8") ctx.fill()
+		core.fillBoldText1(
+			ctx,
+			'全部',
+			170,
+			190,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			'收藏★',
+			170,
+			290,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_1", "默认分类1"),
+			170,
+			390,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_2", "默认分类2"),
+			170,
+			490,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_3", "默认分类3"),
+			170,
+			590,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_4", "默认分类4"),
+			170,
+			690,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_5", "默认分类5"),
+			170,
+			790,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_6", "默认分类6"),
+			170,
+			890,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_7", "默认分类7"),
+			170,
+			990,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.fillBoldText1(
+			ctx,
+			core.getLocalStorage("slClass_8", "默认分类8"),
+			170,
+			1090,
+			'#DDDDDD',
+			"#000000",
+			6,
+			this._buildFont(45, true)
+		);
+		core.strokeRoundRect(ctx, 300, 100, 1648, 1048, 30, "#DDDDDD")
+		// --- 绘制记录
+		this._drawSLPanel_drawRecords();
+		ctx.restore();
+
+
+	}
+
+
+	ui.prototype._drawSLPanel_drawBackground = function () {
+
+		const bg = core.material.images.images["eve_011101.webp"]; //背景
+		core.fillRect(ctx, 0, 0, 2028, 1248, "#000000"); //黑色背景
+		ctx.globalAlpha = 0.5; //透明度
+		if (bg) ctx.drawImage(bg, 0, 0, 1280, 720, 0, 0, 2028, 1248); //绘制半透明背景图片
+		ctx.globalAlpha = 1; //恢复为不透明
+
+	}
+
+	ui.prototype._drawSLPanel_loadSave = function (page, callback) {
+		var ids = [0];
+		for (var i = 1; i <= 10; ++i) {
+			var id = 10 * page + i;
+			if (core.status.event.data.mode == 'fav')
+				id = core.saves.favorite[id - 1]; // 因为favorite第一个不是自动存档 所以要偏移1
+			if (core.status.event.data.mode !== 'fav' && core.status.event.data.mode !== "all")
+				id = core.saves[core.status.event.data.mode][id - 1]
+			ids.push(id);
+		}
+		core.getSaves(ids, function (data) {
+			for (var i = 1; i < ids.length; ++i)
+				core.status.event.ui[i] = data[i];
+			core.status.event.ui[0] = data[0] == null ? null : data[0][core.saves.autosave.now - 1];
+			callback();
+		});
+	}
+
+	// 在以x为中心轴 y为顶坐标 的位置绘制一条宽为size的记录 cho表示是否被选中 选中会加粗 highlight表示高亮标题 ✐
+	ui.prototype._drawSLPanel_drawRecord = function (page, offset) {
+		var data = core.status.event.ui[offset]
+		var id = 9 * page + offset
+		core.drawWindowSkin(
+			"winskin.webp",
+			ctx, 1032,
+			252,
+			660,
+			660,
+			null,
+			null,
+			null,
+			3
+		);
+		if (data && data.floorId) {
+			core.setTextAlign('ui', "center");
+			var map = core.maps.loadMap(data.maps, data.floorId);
+			core.extractBlocksForUI(map, data.hero.flags);
+
+			core.drawThumbnail(data.floorId, map.blocks, {
+				heroLoc: data.hero.loc,
+				heroIcon: data.hero.image,
+				flags: data.hero.flags,
+				ctx: ctx,
+				x: 1050,
+				y: 270,
+				size: 1.5,
+				centerX: data.hero.loc.x,
+				centerY: data.hero.loc.y,
+				noHD: true
+			});
+			if (core.isPlaying() && core.getFlag("hard") != data.hero.flags.hard) {
+				core.fillRect(ctx, 1050, 270, 624, 624, [0, 0, 0, 0.4]);
+				core.fillText(ctx, data.hard, 1362, 620, data.hero.flags.__hardColor__ || 'white', this._buildFont(180, true));
+			}
+			core.drawIcon(ctx, "hp", 800, 260, 96, 96);
+			core.drawIcon(ctx, "atk", 800, 370, 96, 96);
+			core.drawIcon(ctx, "def", 800, 470, 96, 96);
+			core.drawIcon(ctx, "I374", 800, 580, 96, 96);
+			core.drawIcon(ctx, "jumpShoes", 800, 690, 96, 96);
+			core.drawIcon(ctx, "money", 800, 810, 96, 96);
+			core.setTextAlign(ctx, 'left');
+			core.fillBoldText1(
+				ctx,
+				core.formatBigNumber(data.hero.hp, true),
+				900,
+				320,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				core.formatBigNumber(data.hero.atk, true),
+				900,
+				430,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				core.formatBigNumber(data.hero.def, true),
+				900,
+				540,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				core.formatBigNumber(data.hero.spell, true),
+				900,
+				650,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				core.formatBigNumber(data.hero.speed, true),
+				900,
+				760,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				core.formatBigNumber(data.hero.money, true),
+				900,
+				880,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.setTextAlign(ctx, 'center');
+			core.fillBoldText1(
+				ctx,
+				core.floors[data.floorId].title,
+				1350,
+				150,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				`[${core.floors[data.floorId].areas}]`,
+				1350,
+				220,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.drawWindowSkin(
+				"winskin.webp",
+				ctx, 1700,
+				260,
+				96,
+				96,
+				null,
+				null,
+				null,
+				1
+			);
+			core.drawWindowSkin(
+				"winskin.webp",
+				ctx, 1700,
+				400,
+				96,
+				96,
+				null,
+				null,
+				null,
+				1
+			);
+			core.drawWindowSkin(
+				"winskin.webp",
+				ctx, 1700,
+				540,
+				96,
+				96,
+				null,
+				null,
+				null,
+				1
+			);
+			core.drawWindowSkin(
+				"winskin.webp",
+				ctx, 1700,
+				680,
+				96,
+				96,
+				null,
+				null,
+				null,
+				1
+			);
+			core.drawWindowSkin(
+				"winskin.webp",
+				ctx, 1700,
+				810,
+				96,
+				96,
+				null,
+				null,
+				null,
+				1
+			);
+			if (data.hero.equipment[0]) core.drawIcon(ctx, data.hero.equipment[0], 1700, 260, 96, 96)
+			else core.fillBoldText1(
+				ctx,
+				"无",
+				1750,
+				320,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+
+			if (data.hero.equipment[1]) core.drawIcon(ctx, data.hero.equipment[1], 1700, 400, 96, 96)
+			else core.fillBoldText1(
+				ctx,
+				"无",
+				1750,
+				460,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+
+			if (data.hero.equipment[2]) core.drawIcon(ctx, data.hero.equipment[2], 1700, 540, 96, 96)
+			else core.fillBoldText1(
+				ctx,
+				"无",
+				1750,
+				600,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+
+			if (data.hero.equipment[3]) core.drawIcon(ctx, data.hero.equipment[3], 1700, 680, 96, 96)
+			else core.fillBoldText1(
+				ctx,
+				"无",
+				1750,
+				740,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			if (data.hero.equipment[3]) core.drawIcon(ctx, data.hero.equipment[3], 1700, 810, 96, 96)
+			else core.fillBoldText1(
+				ctx,
+				"无",
+				1750,
+				870,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.drawIcon(ctx, "yellowKey", 1800, 260, 96, 96)
+			core.drawIcon(ctx, "blueKey", 1800, 420, 96, 96)
+			core.drawIcon(ctx, "redKey", 1800, 580, 96, 96)
+			core.drawIcon(ctx, "greenKey", 1800, 740, 96, 96)
+			core.fillBoldText1(
+				ctx,
+				data.hero.items.tools.yellowKey ?? 0,
+				1850,
+				390,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				data.hero.items.tools.blueKey ?? 0,
+				1850,
+				550,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				data.hero.items.tools.redKey ?? 0,
+				1850,
+				710,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				data.hero.items.tools.greenKey ?? 0,
+				1850,
+				870,
+				'#DDDDDD',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			core.fillBoldText1(
+				ctx,
+				`存档时间${core.formatDate(new Date(data.time))}`,
+				1350, 960, data.hero.flags.debug ? '#FF6A6A' : '#FFFFFF',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			if (offset !== 0) core.fillBoldText1(
+				ctx,
+				"修改存档名",
+				920, 1020, '#FFFFFF',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			if (offset !== 0) core.fillBoldText1(
+				ctx,
+				"存档分类",
+				1800, 1020, '#FFFFFF',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			if (offset !== 0) core.fillBoldText1(
+				ctx,
+				"删除存档",
+				1800, 1120, '#FFFFFF',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+
+			if (core.status.event.data.mode !== "all" && core.status.event.data.mode !== "fav") core.fillBoldText1(
+				ctx,
+				"修改分类名",
+				920, 1120, '#FFFFFF',
+				"#000000",
+				6,
+				this._buildFont(45, true)
+			);
+			if (data.hero.flags._cgText) {
+				core.fillBoldText1(
+					ctx,
+					cg ? "剧情CG模式" : "缩略图模式",
+					1350, 1220, '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(45, true)
+				);
+				if (!cg) {
+					core.drawWindowSkin(
+						"winskin.webp",
+						ctx, 805,
+						252,
+						1140,
+						660,
+						null,
+						null,
+						null,
+						3
+					);
+					core.drawImage(ctx, data.cg.image, 823, 270, 1104, 624)
+					let s = 0.5
+					data.cg.bodyList.forEach((v) => {
+						//绘制立绘
+						const body = core.material.images.images?.[v.name];
+						if (v.filter) ctx.filter = "brightness(50%)";
+
+						if (body) {
+							if (!v.w && !v.h && !v.scale) v.scale = 1.7;
+
+							if (!v.w && !v.h) {
+								ctx.drawImage(
+									body,
+									0,
+									0,
+									body.width,
+									body.height,
+									823 + v.px * s,
+									894 - body.height * v.scale * s,
+									body.width * v.scale * s,
+									body.height * v.scale * s
+								);
+							} else {
+								ctx.drawImage(
+									body,
+									0,
+									0,
+									body.width,
+									body.height,
+									823 + v.px * s,
+									894 - (v.h ?? body.height) * s,
+									v.w ?? body.width * s,
+									v.h ?? body.height * s
+								);
+							}
+						}
+						ctx.filter = "none";
+					});
+					core.fillRect(ctx, 823, 694, 1104, 200, "rgba(0,0,0,0.3)")
+					const head = core.material.images.images?.[data.cg.head.name];
+					if (head) {
+						//绘制头像
+						ctx.drawImage(
+							head,
+							0,
+							0,
+							head.width,
+							head.height,
+							823 + data.cg.head.px * s,
+							894 - head.height * 2.2 * s,
+							head.width * 2.2 * s,
+							head.height * 2.2 * s
+						);
+
+
+					}
+					core.setTextAlign(ctx, "left")
+					if (data.cg.name) core.fillBoldText1(
+						ctx,
+						`【${data.cg.name}】`,
+						1143, 720, '#FFFFFF',
+						"#000000",
+						2,
+						this._buildFont(30, true)
+					);
+					if (data.cg.text) core.drawTextContent(ctx, data.cg.text, {
+						left: 1143,
+						top: 750,
+						bold: true,
+						color: "#FFFFFF",
+						align: "left",
+						fontSize: 30,
+						time: 0,
+						font: "Verdana",
+						maxWidth: 800,
+					});
+				}
+			}
+			if (offset !== 0) core.drawWindowSkin(
+				"winskin.webp",
+				ctx, 1050,
+				982,
+				640,
+				150,
+				null,
+				null,
+				null,
+				1
+			);
+			let real_id = id
+			if (core.status.event.data.mode === "fav") {
+				real_id = core.saves.favorite[id - 1]
+			}
+			if (core.status.event.data.mode !== "fav" && core.status.event.data.mode !== "all") {
+				real_id = core.saves[core.status.event.data.mode][id - 1]
+			}
+			if (offset !== 0) core.drawTextContent(ctx, core.getLocalStorage("_slText" + real_id, "点击填写存档笔记"), {
+				left: 1070,
+				top: 1000,
+				bold: true,
+				color: "#FFFFFF",
+				align: "left",
+				fontSize: 30,
+				time: 0,
+				font: "HATTEN",
+				maxWidth: 620,
+			});
+			if (choose) {
+				core.drawWindowSkin(
+					"winskin.webp",
+					ctx, 805,
+					252,
+					1140,
+					660,
+					null,
+					null,
+					null,
+					3
+				);
+				core.fillRect(ctx, 823, 270, 1104, 624, "#000000")
+
+				core.setTextAlign(ctx, 'left')
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.favorite.indexOf(real_id)>=0?"☑":"□"}收藏★`,
+					920, 370, core.saves.favorite.indexOf(real_id) >= 0 ? '#FFFF44' : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_1.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_1", "默认分类1")}`,
+					920, 480, core.saves.class_1.indexOf(real_id) >= 0 ? '#FFFF44' : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_2.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_2", "默认分类2")}`,
+					920, 590, core.saves.class_2.indexOf(real_id) >= 0 ? "#FFFF44" : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_3.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_3", "默认分类3")}`,
+					920, 700, core.saves.class_3.indexOf(real_id) >= 0 ? "#FFFF44" : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_4.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_4", "默认分类4")}`,
+					920, 810, core.saves.class_4.indexOf(real_id) >= 0 ? "#FFFF44" : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_5.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_5", "默认分类5")}`,
+					1400, 370, core.saves.class_5.indexOf(real_id) >= 0 ? "#FFFF44" : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_6.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_6", "默认分类6")}`,
+					1400, 480, core.saves.class_6.indexOf(real_id) >= 0 ? "#FFFF44" : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_7.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_7", "默认分类7")}`,
+					1400, 590, core.saves.class_7.indexOf(real_id) >= 0 ? "#FFFF44" : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					`${core.saves.class_8.indexOf(real_id)>=0?"☑":"□"}${core.getLocalStorage("slClass_8", "默认分类8")}`,
+					1400, 700, core.saves.class_8.indexOf(real_id) >= 0 ? "#FFFF44" : '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+				core.fillBoldText1(
+					ctx,
+					"返回",
+					1700, 810, '#FFFFFF',
+					"#000000",
+					6,
+					this._buildFont(60, true)
+				);
+
+			}
+			ctx.lineWidth = 6
+			core.setTextAlign(ctx, "center")
+		} else {
+
+			core.fillRect(ctx, 1050, 270, 624, 624, '#333333');
+			core.fillText(ctx, '无存档', 1362, 620, "#444444", this._buildFont(180, true));
+		}
+	}
+
+	ui.prototype._drawSLPanel_drawRecords = function (n) {
+		var page = core.status.event.data.page;
+		var offset = core.status.event.data.offset;
+		var name = core.status.event.id == 'save' ? "存档" : core.status.event.id == 'load' ? "读档" : "回放";
+		core.drawLine(ctx, 650, 100, 650, 1148)
+		core.drawLine(ctx, 800, 100, 800, 1148)
+		let liney = 0
+		core.fillText(
+			ctx,
+			"自动存档",
+			470,
+			190,
+			offset === 0 ? '#DDDD44' : core.status.event.ui[0] ? "#FFFFFF" : "#BBBBBB",
+			this._buildFont(45, true)
+		);
+		core.setStrokeStyle(ctx, "#DDDDDD")
+		core.drawLine(ctx, 300, 224, 650, 224)
+		core.fillText(
+			ctx,
+			"▲",
+			725,
+			240,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▲",
+			725,
+			220,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▲",
+			725,
+			200,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▲",
+			725,
+			320,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▲",
+			725,
+			340,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▲",
+			725,
+			440,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			page + 1,
+			725,
+			540,
+			"#DDDD44",
+			this._buildFont(45, true)
+		);
+		core.fillText(
+			ctx,
+			"/",
+			725,
+			590,
+			"#DDDDDD",
+			this._buildFont(45, true)
+		);
+		core.fillText(
+			ctx,
+			core.saveLoad.max_page,
+			725,
+			640,
+			"#DDDDDD",
+			this._buildFont(45, true)
+		);
+		core.fillText(
+			ctx,
+			"▼",
+			725,
+			740,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▼",
+			725,
+			820,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▼",
+			725,
+			840,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▼",
+			725,
+			940,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▼",
+			725,
+			960,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		core.fillText(
+			ctx,
+			"▼",
+			725,
+			980,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+		if (core.status.event.data.mode === 'all' && core.getLocalStorage("save_page", 0) !== 8000) core.fillText(
+			ctx,
+			"扩容",
+			725,
+			1090,
+			"#DDDDDD",
+			this._buildFont(60, true)
+		);
+
+		this._drawSLPanel_drawRecord(page, offset)
+		for (var i = 1; i < (n || 10); i++) {
+			var data = core.status.event.ui[i];
+			var id = 9 * page + i;
+			var highLight = (core.status.event.data.mode !== 'fav' && core.saves.favorite.indexOf(id) >= 0) || core.status.event.data.mode == 'fav';
+			var title = (highLight ? '★ ' : '☆ ') + (core.saves.favoriteName[id] || (name + id));
+			if (i != 0 && core.status.event.data.mode !== 'all') {
+				if (!data) break;
+				var real_id = core.saves.favorite[id - 1];
+				for (let a = 1; a < 9; a++) {
+					if (core.status.event.data.mode === ("class_" + a))
+						real_id = core.saves["class_" + a][id - 1]
+				}
+				title = (core.saves.favoriteName[real_id] || (name + real_id));
+
+			}
+			if (data && data.__toReplay__) title = '[R]' + title;
+			liney += 100
+			core.fillText(
+				ctx,
+				title,
+				470,
+				190 + liney,
+				offset === i ? '#DDDD44' : data ? "#FFFFFF" : "#BBBBBB",
+				this._buildFont(45, true)
+			);
+			core.setStrokeStyle(ctx, "#DDDDDD")
+			if (i != 9) core.drawLine(ctx, 300, 224 + liney, 650, 224 + liney)
+
+		}
+	};
+	////// 点击保存按钮时的打开操作 //////
+	events.prototype.save = function (fromUserAction) {
+		if (core.isReplaying()) return;
+		if (core.hasFlag("__forbidSave__")) {
+			core.playSound("操作失败");
+			core.drawTip("当前禁止存档");
+			return;
+		}
+		if (
+			core.status.event.id == "save" &&
+			core.events.recoverEvents(core.status.event.interval)
+		)
+			return;
+		if (!this._checkStatus("save", fromUserAction)) return;
+		var saveIndex = core.saves.saveIndex;
+		var page = parseInt((saveIndex - 1) / 9),
+			offset = saveIndex - 9 * page;
+		core.playSound("打开界面");
+		core.ui._drawSLPanel(10 * page + offset);
+	};
+
+	////// 点击读取按钮时的打开操作 //////
+	events.prototype.load = function (fromUserAction) {
+		if (core.isReplaying()) return;
+		var saveIndex = core.saves.saveIndex;
+		var page = parseInt((saveIndex - 1) / 9),
+			offset = saveIndex - 9 * page;
+		// 游戏开始前读档
+		if (!core.isPlaying()) {
+			core.dom.startPanel.style.display = "none";
+			core.clearStatus();
+			core.clearMap("all");
+			core.status.event = { id: "load", data: null };
+			core.status.lockControl = true;
+			core.playSound("打开界面");
+			core.ui._drawSLPanel(10 * page + offset);
+			return;
+		}
+		if (
+			core.status.event.id == "load" &&
+			core.events.recoverEvents(core.status.event.interval)
+		)
+			return;
+		if (!this._checkStatus("load", fromUserAction)) return;
+		core.playSound("打开界面");
+		core.ui._drawSLPanel(10 * page + offset);
+	};
+	////// 回放时存档 //////
+	control.prototype._replay_SL = function () {
+		if (!core.isPlaying() || !core.isReplaying()) return;
+		if (!core.status.replay.pausing) {
+			core.playSound("操作失败");
+			return core.drawTip("请先暂停录像");
+		}
+		if (core.isMoving() || core.status.replay.animate || core.status.event.id) {
+			core.playSound("操作失败");
+			return core.drawTip("请等待当前事件的处理结束");
+		}
+		if (core.hasFlag("__forbidSave__")) {
+			core.playSound("操作失败");
+			return core.drawTip("当前禁止存档");
+		}
+		this._replay_hideProgress();
+
+		core.lockControl();
+		core.status.event.id = "save";
+		var saveIndex = core.saves.saveIndex;
+		var page = parseInt((saveIndex - 1) / 9),
+			offset = saveIndex - 9 * page;
+
+		core.ui._drawSLPanel(10 * page + offset);
+	};
+
+
+
+	////// 存读档界面时,按下某个键的操作 //////
+	actions.prototype._keyDownSL = function (keycode) {
+
+		var page = core.status.event.data.page,
+			offset = core.status.event.data.offset;
+		var index = page * 10 + offset;
+
+		if (keycode == 37) { // left
+			core.playSound('光标移动');
+
+			core.ui._drawSLPanel(10 * (page - 1) + offset);
+			return;
+		}
+		if (keycode == 38) { // up
+			core.playSound('光标移动');
+			if (offset > 1) {
+				core.ui._drawSLPanel(10 * page + offset - 1);
+			} else {
+				core.ui._drawSLPanel(10 * (page - 1) + 9);
+			}
+			return;
+		}
+		if (keycode == 39) { // right
+			core.playSound('光标移动');
+
+			core.ui._drawSLPanel(10 * (page + 1) + offset);
+
+			return;
+		}
+		if (keycode == 40) { // down
+			core.playSound('光标移动');
+			if (offset < 9) {
+				core.ui._drawSLPanel(10 * page + offset + 1);
+			} else {
+				core.ui._drawSLPanel(10 * (page + 1));
+			}
+			return;
+		}
+		if (keycode == 33) { // PAGEUP
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page - 10) + offset);
+			return;
+		}
+		if (keycode == 34) { // PAGEDOWN
+			core.playSound('光标移动');
+			core.ui._drawSLPanel(10 * (page + 10) + offset);
+			return;
+		}
+	}
+
+	////// 存读档界面时,放开某个键的操作 //////
+	actions.prototype._keyUpSL = function (keycode) {
+		var page = core.status.event.data.page,
+			offset = core.status.event.data.offset;
+		var index = page * 10 + offset;
+
+		if (keycode == 27 || keycode == 88 || (core.status.event.id == 'save' && keycode == 83) || (core.status.event.id == 'load' && keycode == 68)) { //退出
+			saveLoadclick(1800, 1200);
+			return;
+		}
+		if (keycode >= 48 && keycode <= 57) {
+			if (keycode == 48) keycode = 58;
+			core.ui._drawSLPanel((keycode - 49) * 1000 + 1);
+			return;
+		}
+		if (keycode == 13 || keycode == 32 || keycode == 67) {
+			if (offset == 0)
+				core.doSL("autoSave", core.status.event.id);
+			else {
+				var id = 9 * page + offset;
+				if (core.status.event.data.mode == 'fav') id = core.saves.favorite[id - 1];
+				if (core.status.event.data.mode !== 'fav' && core.status.event.data.mode !== 'all') id = core.saves[core.status.event.data.mode][id - 1];
+				core.doSL(id, core.status.event.id);
+			}
+			return;
+		}
+		if (keycode == 69) { //e
+			switch (core.status.event.data.mode) {
+			case "all":
+				core.status.event.data.mode = 'fav'
+				break;
+			case "fav":
+				core.status.event.data.mode = 'class_1'
+				break;
+			case "class_1":
+				core.status.event.data.mode = 'class_2'
+				break;
+			case "class_2":
+				core.status.event.data.mode = 'class_3'
+				break;
+			case "class_3":
+				core.status.event.data.mode = 'class_4'
+				break;
+			case "class_4":
+				core.status.event.data.mode = 'class_5'
+				break;
+			case "class_5":
+				core.status.event.data.mode = 'class_6'
+				break;
+			case "class_6":
+				core.status.event.data.mode = 'class_7'
+				break;
+			case "class_7":
+				core.status.event.data.mode = 'class_8'
+				break;
+			case "class_8":
+				core.status.event.data.mode = 'all'
+				break;
+			}
+
+			core.ui._drawSLPanel(0)
+		}
+		if (keycode == 46) {
+			if (offset == 0) {
+				core.playSound('操作失败');
+				core.drawTip("无法删除自动存档!");
+			} else {
+				var id = 9 * page + offset;
+				if (core.status.event.data.mode == 'fav') id = core.saves.favorite[id - 1];
+				if (core.status.event.data.mode !== 'fav' && core.status.event.data.mode !== 'all') id = core.saves[core.status.event.data.mode][id - 1];
+				core.removeSave(id, function () {
+					core.ui._drawSLPanel(index, true);
+				});
+			}
+		}
+
+	}
+},
+    "scrollingText": function () {
+	// 本插件用于绘制在线留言
+	// 说明:https://h5mota.com/bbs/thread/?tid=1017
+	// 目前使用core.http代替帖子中提到的axios
+	////// 绘制提示同时播放成功音效 //////
+	ui.prototype.drawSuccessTip = function (text, id, frame) {
+		this.drawTip(text, id, frame);
+		core.playSound('gem.mp3');
+	}
+	core.plugin.aniMap = new Map()
+	////// 绘制提示同时播放错误音效 //////
+	ui.prototype.drawFailTip = function (text, id, frame) {
+		this.drawTip(text, id, frame);
+		core.playSound('error.mp3');
+	}
+	/** 塔的英文名 */
+	const towerName = core.firstData.name;
+
+	let [W, H] = [core.__SIZE__, core.__SIZE__];
+	let [WIDTH, HEIGHT] = [core.__PIXELS__, core.__PIXELS__];
+
+	//#region 弹幕的收发
+	this.getComment = function () {
+		if (core.isReplaying()) return;
+		let form = new FormData();
+		form.append('type', 1);
+		form.append('towername', towerName);
+		utils.prototype.http(
+			'POST',
+			'https://h5mota.com/backend/tower/barrage.php',
+			form,
+			function (res) {
+				try {
+					res = JSON.parse(res);
+					console.log(res);
+					core.drawTip('接收成功!')
+					core.playSound('item.mp3');
+					let commentCollection = {};
+					const commentList = res?.list;
+					const isEmpty = /^\s*$/;
+					for (let i = 0, l = commentList.length; i <= l - 1; i++) {
+						if (isEmpty.test(commentList[i]?.comment)) continue;
+						const commentTagsList = commentList[i].tags.split(',');
+						const [cFloorId, cX, cY] = commentTagsList;
+						if (0 <= cX && cX <= W - 1 && 0 <= cY && cY <= H - 1 && core.floorIds.includes(cFloorId)) {
+							if (!commentCollection.hasOwnProperty(cFloorId)) { commentCollection[cFloorId] = {}; }
+							const str = cX + ',' + cY;
+							if (!commentCollection[cFloorId].hasOwnProperty(str)) { commentCollection[cFloorId][str] = []; }
+							commentCollection[cFloorId][str].push(commentList[i]?.comment);
+						}
+					}
+					core.setFlag('commentCollection', commentCollection);
+				} catch (err) {
+					core.drawFailTip('接收失败!' + err.message);
+				}
+			},
+			function (err) {
+				err = JSON.parse(err);
+				console.error(err);
+				core.drawFailTip('接收失败' + err?.message);
+			},
+			null, null, null, 1000
+		);
+	}
+
+	this.postComment = function (comment, tags) {
+		if (core.isReplaying()) return;
+		const isEmpty = /^\s*$/;
+		if (isEmpty.test(comment)) {
+			core.drawFailTip('您输入的消息为空,请重发!');
+			return;
+		}
+		let form = new FormData();
+		form.append('type', 2);
+		form.append('towername', towerName);
+		form.append('comment', comment);
+		form.append('tags', tags);
+		form.append('userid', 2324);
+		form.append('password', '77c8fd5ff49c370342e4472ebdda5903');
+		utils.prototype.http(
+			'POST',
+			'https://h5mota.com/backend/tower/barrage.php',
+			form,
+			function (res) {
+				try {
+					res = JSON.parse(res);
+					console.log(res);
+					if (res?.code === 0) {
+						core.drawTip('提交成功!')
+					} else {
+						core.drawTip('提交失败!' + res?.message);
+					}
+				} catch (err) {
+					core.drawFailTip('提交失败!' + err.message);
+				}
+			},
+			function (err) {
+				err = JSON.parse(err);
+				console.error(err);
+				core.drawFailTip('提交失败!' + err?.message);
+			},
+			null, null, null, 1000
+		);
+	}
+	//#endregion
+
+	/** 若变量comment为真,在每层切换时在地上有弹幕的地方显示相应图标。 */
+	this.drawCommentSign = function () {
+		if (!core.hasFlag('comment') || core.isReplaying()) return;
+		let commentCollection = core.getFlag('commentCollection', {}),
+			floorId = core.status.floorId;
+		core.createCanvas('sign', 0, 0, WIDTH, HEIGHT, 61);
+		core.setOpacity('sign', 0.6);
+		if (commentCollection.hasOwnProperty(floorId)) {
+			for (let pos in commentCollection[floorId]) {
+				const l = commentCollection[floorId][pos].length;
+				for (let i = 0; i <= l - 1; i++) {
+					const [x, y] = pos.split(',');
+					core.drawIcon('sign', 'postman', 32 * x, 32 * y);
+					break;
+				}
+			}
+		}
+	}
+
+	/** 立即清除楼层的弹幕图标。关闭弹幕相关设置时调用。 */
+	this.clearCommentSign = function () {
+		core.deleteCanvas('sign');
+	}
+
+	/** 默认一次显示的弹幕数 */
+	const showNum = 5;
+
+	// 每走一步或瞬移,调用该函数,若目标点有弹幕,显示之
+	this.showComment = function (x, y) {
+		if (!core.getFlag('comment') || core.isReplaying()) return;
+		const commentCollection = core.getFlag('commentCollection', {});
+		const floorId = core.status.floorId,
+			str = x + ',' + y;
+		if (commentCollection.hasOwnProperty(floorId) &&
+			commentCollection[floorId].hasOwnProperty(str)) {
+			let commentArr = commentCollection[floorId][str].concat();
+			const commentArrPicked = pickComment(commentArr, showNum);
+			drawComment(commentArrPicked);
+		}
+	}
+
+	/** 返回从commentArr中挑选showNum个comment组成的数组*/
+	function pickComment(commentArr, showNum) {
+		let showList = [];
+		if (commentArr.length <= showNum) {
+			showList = commentArr;
+		} else {
+			for (let i = 0; i <= showNum - 1; i++) {
+				const l = commentArr.length,
+					n = core.plugin.dice(l - 1);
+				showList.push(commentArr[n]);
+				commentArr.splice(n, 1);
+			}
+		}
+		return showList;
+	}
+
+	function drawComment(commentArr) {
+		const l = commentArr.length;
+		let yList = generateCommentYList(20, HEIGHT - 20, showNum);
+		if (l < showNum) yList = getRandomElements(yList, l);
+		for (let i = 0; i <= l - 1; i++) {
+			drawCommentStr(commentArr[i], WIDTH + 20 * Math.random(),
+				yList[i], Math.random() * 0.1 + 0.1);
+		}
+	}
+
+	/** 生成count个随机数,范围从min到max,作为弹幕的y坐标*/
+	function generateCommentYList(min, max, count) {
+		let yList = Array(count).fill(0);
+		const distance = (max - min) / (count + 1);
+		for (let i = 0; i < count; i++) {
+			yList[i] = min + distance * (i + 1) + (Math.random() - 0.5) * (distance / 2);
+		}
+		return yList;
+	}
+
+	function getRandomElements(arr, count) {
+		let result = [...arr];
+		let len = result.length;
+		count = Math.min(len, count);
+
+		for (let i = len - 1; i > len - 1 - count; i--) {
+			let j = Math.floor(Math.random() * (i + 1));
+			[result[i], result[j]] = [result[j], result[i]];
+		}
+
+		return result.slice(len - count);
+	}
+
+	//#region 弹幕绘制部分
+	const { Animation, linear, Ticker } = core.plugin.animate ?? {};
+	const ctxName = 'scrollingText';
+
+	if (Ticker) {
+		const ticker = new Ticker();
+		ticker.add(() => {
+			if (core.isReplaying()) return;
+			core.createCanvas(ctxName, 0, 0, core.__PIXELS__, core.__PIXELS__, 136); //每帧重绘该画布
+		});
+	}
+
+	/**
+	 * 绘制弹幕 
+	 * @example  
+	 * drawCommentStr('OK', 450, 200, 0.1);
+	 * @param {string} content 弹幕的内容
+	 * @param {number} x 弹幕的初始x坐标
+	 * @param {number} y 弹幕的初始y坐标
+	 * @param {number} vx 弹幕的横向滚动速度
+	 */
+	function drawCommentStr(content, x, y, vx) {
+		if (core.isReplaying() || !Animation) return;
+		const ani = new Animation();
+		core.plugin.aniMap.set(ani, (ani) => ani.ticker.destroy());
+		ani.ticker.add(() => {
+			core.fillText(ctxName, content, x + ani.x, y, 'white', '16px Verdana');
+		})
+		// 弹幕的最大长度5600,再长属于异常数据
+		const aim = 600 + Math.min(core.calWidth(ctxName, content, '16px Verdana'), 5000);
+		ani.mode(linear())
+			.time(aim / vx)
+			.absolute()
+			.move(-aim, 0)
+		ani.all().then(() => {
+			ani.ticker.destroy();
+		});
+	}
+	//#endregion 
+
+},
+    "动画": function () {
+	// 在此增加新插件
+	const { Transition, linear, bezier, circle, hyper, trigo, power, inverseTrigo, shake, sleep } = core.plugin.animate;
+	//////移动动画//////
+	const tran = new Transition();
+	maps.prototype.moveAnimate = function (id, px, py, relative, time, style) {
+		if (!id) return
+		core.status.animateObjs.forEach(v => {
+			if (v.id === id) {
+				if (v.hero) return
+				if (!style) tran.mode(linear())
+				else if (style === "in") {
+					tran.mode(trigo('sin', "in"))
+				} else if (style === "out") {
+					tran.mode(trigo('sin', "out"))
+				} else if (style === "in-out") {
+					tran.mode(trigo('sin', "in-out"))
+				} else if (style === "center") {
+					tran.mode(trigo('sin', "center"))
+				}
+				if (relative) { tran.relative() } else { tran.absolute() }
+				tran.time(time)
+				tran.value[v.id + v.name + "x"] = px
+				tran.value[v.id + v.name + "y"] = py
+
+			}
+		})
+	}
+
+	control.prototype._animationFrame_animate = function (timestamp) {
+		let frametime = timestamp - core.animateFrame.animateTime
+		core.animateFrame.animateTime = timestamp;
+		if (!core.status.animateObjs || core.status.animateObjs.length == 0) return;
+
+		core.clearMap('animate');
+		// 更新帧
+		for (var i = 0; i < core.status.animateObjs.length; i++) {
+			var obj = core.status.animateObjs[i];
+			if (obj.pause) obj.pausetime += frametime
+			if (obj.start === 0) obj.start = core.animateFrame.animateTime
+			obj.index = Math.floor((core.animateFrame.animateTime - obj.start - obj.pausetime) / (1000 / 60))
+			if (obj.reverse) obj.index = obj.animate.frames.length - obj.index
+			if ((!obj.reverse && obj.index >= obj.animate.frames.length) || (obj.reverse && obj.index <= 0)) {
+				if (obj.loop) {
+					if (obj.reverse) {
+						obj.index = obj.animate.frames.length
+						obj.start = core.animateFrame.animateTime
+						obj.pausetime = 0
+					} else {
+						obj.start = core.animateFrame.animateTime
+						obj.index = 0
+						obj.pausetime = 0
+					}
+				} else
+					(function (callback) {
+						setTimeout(function () {
+							if (callback) callback();
+						});
+					})(obj.callback);
+			}
+		}
+		core.status.animateObjs = core.status.animateObjs.filter(function (obj) {
+			return (!obj.reverse && obj.index < obj.animate.frames.length) || (obj.reverse && obj.index > 0);
+		});
+		let sx, sy;
+		if (core.status.heroMoving < 0) {
+			sx = 0;
+			sy = 0;
+		} else {
+			sx =
+				core.utils.scan[core.status.hero.loc.direction].x *
+				4 *
+				core.status.heroMoving;
+			sy =
+				core.utils.scan[core.status.hero.loc.direction].y *
+				4 *
+				core.status.heroMoving;
+		}
+		const herox = core.status.hero.loc.x * 32 + 16 + sx;
+		const heroy = core.status.hero.loc.y * 32 + 32 + sy - core.material.icons.hero.height / 2;
+
+		core.status.animateObjs.forEach(function (obj) {
+			if (obj.hero) {
+
+				core.maps._drawAnimateFrame('animate', obj.animate, herox, heroy, obj.index);
+			} else {
+
+				core.maps._drawAnimateFrame('animate', obj.animate, tran.value[obj.id + obj.name + "x"], tran.value[obj.id + obj.name + "y"], obj.index);
+			}
+		});
+	}
+	core.registerAnimationFrame("animate", true, core.control._animationFrame_animate);
+
+	////// 绘制动画 //////
+	maps.prototype.drawAnimate = function (name, x, y, alignWindow, callback) {
+		name = core.getMappedName(name);
+
+		// 正在播放录像:不显示动画
+		if (core.isReplaying() || !core.material.animates[name] || x == null || y == null) {
+			if (callback) callback();
+			return -1;
+		}
+
+		// 开始绘制
+		var animate = core.material.animates[name],
+			centerX = 32 * x + 16,
+			centerY = 32 * y + 16;
+		if (alignWindow) {
+			centerX += core.bigmap.offsetX;
+			centerY += core.bigmap.offsetY;
+		}
+		animate.se = animate.se || {};
+		if (typeof animate.se == 'string') animate.se = { 1: animate.se };
+		tran.mode(linear()).time(1).absolute()
+		tran.value[id + name + "x"] = centerX
+		tran.value[id + name + "y"] = centerY
+		var id = setTimeout(null);
+		core.status.animateObjs.push({
+			"name": name,
+			"id": id,
+			"animate": animate,
+			"centerX": centerX,
+			"centerY": centerY,
+			"start": 0,
+			"pause": false,
+			"pausetime": 0,
+			"index": 0,
+			"callback": callback
+		});
+
+		return id;
+	}
+
+	////// 绘制反转动画 //////
+	maps.prototype.drawResizeAnimate = function (name, id, centerX, centerY, hero, reverse, loop, callback) {
+		name = core.getMappedName(name);
+
+		// 正在播放录像:不显示动画
+		if (core.isReplaying() || !core.material.animates[name]) {
+			if (callback) callback();
+			return -1;
+		}
+
+		// 开始绘制
+		var animate = core.material.animates[name],
+			centerX = core.calValue(centerX)
+		centerY = core.calValue(centerY)
+
+		animate.se = animate.se || {};
+		if (typeof animate.se == 'string') animate.se = { 1: animate.se };
+
+		var id = id || setTimeout(null);
+		tran.mode(linear()).time(1).absolute()
+		if (!hero) tran.value[id + name + "x"] = centerX
+		if (!hero) tran.value[id + name + "y"] = centerY
+		if (hero) core.status.animateObjs.push({
+			"name": name,
+			"id": id,
+			"animate": animate,
+			"hero": true,
+			"reverse": reverse,
+			"start": 0,
+			"pause": false,
+			"loop": loop,
+			"pausetime": 0,
+			"index": 0,
+			"callback": callback
+		})
+		else core.status.animateObjs.push({
+			"name": name,
+			"id": id,
+			"animate": animate,
+			"centerX": centerX,
+			"centerY": centerY,
+			"reverse": reverse,
+			"start": 0,
+			"pause": false,
+			"loop": loop,
+			"pausetime": 0,
+			"index": 0,
+			"callback": callback
+		});
+
+		return id;
+	}
+	events.prototype._action_animateResize = function (data, x, y, prefix) {
+		core.events.__action_doAsyncFunc(data.async, core.maps.drawResizeAnimate, data.name, data.id, data.centerX, data.centerY, data.hero, data.reverse, data.loop);
+	};
+	////// 绘制一个跟随勇士的动画 //////
+	maps.prototype.drawHeroAnimate = function (name, callback) {
+		name = core.getMappedName(name);
+
+		// 正在播放录像或动画不存在:不显示动画
+		if (core.isReplaying() || !core.material.animates[name]) {
+			if (callback) callback();
+			return -1;
+		}
+
+		// 开始绘制
+		var animate = core.material.animates[name];
+		animate.se = animate.se || {};
+		if (typeof animate.se == 'string') animate.se = { 1: animate.se };
+
+		var id = setTimeout(null);
+		core.status.animateObjs.push({
+			"name": name,
+			"id": id,
+			"animate": animate,
+			"hero": true,
+			"start": 0,
+			"pause": false,
+			"pausetime": 0,
+			"index": 0,
+			"callback": callback
+		});
+
+		return id;
+	}
+
+	////// 获得当前正在播放的所有(指定)动画的id列表 //////
+	maps.prototype.getPlayingAnimates = function (name) {
+		return (core.status.animateObjs || []).filter(function (one) {
+			return name == null || one.name == name;
+		}).map(function (one) { return one.id });
+	}
+
+
+	////// 绘制动画的某一帧 //////
+	maps.prototype._drawAnimateFrame = function (name, animate, centerX, centerY, index) {
+		var ctx = core.getContextByName(name);
+		if (!ctx) return;
+		var frame = animate.frames[index % animate.frame];
+		core.playSound((animate.se || {})[index % animate.frame + 1], (animate.pitch || {})[index % animate.frame + 1]);
+		var ratio = animate.ratio;
+		frame.forEach(function (t) {
+			var image = animate.images[t.index];
+			if (!image) return;
+
+			var realWidth = image.width * ratio * t.zoom / 100;
+			var realHeight = image.height * ratio * t.zoom / 100;
+			core.setAlpha(ctx, t.opacity / 255);
+
+			var cx = centerX + t.x,
+				cy = centerY + t.y;
+
+			var ix = cx - realWidth / 2 - core.bigmap.offsetX,
+				iy = cy - realHeight / 2 - core.bigmap.offsetY;
+
+			var mirror = t.mirror ? 'x' : null;
+			var angle = t.angle ? -t.angle * Math.PI / 180 : null;
+			core.drawImage(ctx, image, ix, iy, realWidth, realHeight, null, null, null, null, angle, mirror);
+
+			core.setAlpha(ctx, 1);
+		})
+	}
+	////// 暂停动画 //////
+	maps.prototype.pauseAnimate = function (id) {
+		for (var i = 0; i < core.status.animateObjs.length; i++) {
+			var obj = core.status.animateObjs[i];
+			if (id == null || obj.id == id) {
+				obj.pause = true
+
+			}
+		}
+
+	}
+	////// 继续动画 //////
+	maps.prototype.remuseAnimate = function (id) {
+		for (var i = 0; i < core.status.animateObjs.length; i++) {
+			var obj = core.status.animateObjs[i];
+			if (id == null || obj.id == id) {
+				obj.pause = false
+			}
+
+		}
+
+	}
+	////// 停止动画 //////
+	maps.prototype.stopAnimate = function (id, doCallback) {
+		for (var i = 0; i < core.status.animateObjs.length; i++) {
+			var obj = core.status.animateObjs[i];
+			if (id == null || obj.id == id) {
+				if (doCallback) {
+					(function (callback) {
+						setTimeout(function () {
+							if (callback) callback();
+						});
+					})(obj.callback);
+				}
+			}
+		}
+		core.status.animateObjs = core.status.animateObjs.filter(function (x) { return id != null && x.id != id });
+		if (core.status.animateObjs.length == 0)
+			core.clearMap('animate');
+	}
+},
+    "帧动画(游戏画面)": function () {
+	// 在此增加新插件
+	// 在此增加新插件
+	const animate3 = document.createElement("canvas"); //画布设置
+	animate3.style.zIndex = 71;
+	animate3.id = "animate3";
+	animate3.classList.add("gameCanvas", "anti-aliasing");
+	animate3.style.display = "block";
+	animate3.width = 416;
+	animate3.height = 416;
+	animate3.style.width = core.__PIXELS__ * core.domStyle.scale + "px";
+	animate3.style.height = core.__PIXELS__ * core.domStyle.scale + "px";
+	main.dom.animate3 = animate3;
+	const anctx = animate3.getContext("2d");
+
+	main.dom.gameDraw.appendChild(animate3);
+
+	core.plugin.playing = new Set();
+	const { Transition, linear, bezier, circle, hyper, trigo, power, inverseTrigo, shake, sleep } = core.plugin.animate;
+	const tran = new Transition();
+	this.animatemove = function (id, px, py, relative, time, style) {
+		if (!id) return
+		core.plugin.playing.forEach(v => {
+			if (v.id === id) {
+				if (v.hero) return
+				if (!style) tran.mode(linear())
+				else if (style === "in") {
+					tran.mode(trigo('sin', "in"))
+				} else if (style === "out") {
+					tran.mode(trigo('sin', "out"))
+				} else if (style === "in-out") {
+					tran.mode(trigo('sin', "in-out"))
+				} else if (style === "center") {
+					tran.mode(trigo('sin', "center"))
+				}
+				if (relative) { tran.relative() } else { tran.absolute() }
+				tran.time(time)
+				tran.value[v.id + v.name + "x"] = px
+				tran.value[v.id + v.name + "y"] = py
+
+			}
+		})
+	}
+	this.setanimate = function (
+		name,
+		px,
+		py,
+		width,
+		height,
+		allFarme,
+		imageList,
+		soundList
+	) {
+		const data = {
+			px: px,
+			py: py,
+			width: width,
+			height: height,
+			allFarme: allFarme,
+			imageList: imageList,
+			soundList: soundList,
+		};
+		core.setFlag("animate_" + name, data);
+	};
+	this.deleteanimate = function (name) {
+		core.setFlag("animate_" + name);
+	};
+	let thistime = 0;
+	this.playanimate = function (name, id, x, y, hero, scalex, scaley, loop, reverse) {
+		if (!id) id = setTimeout(null)
+		tran.mode(linear()).time(1).absolute()
+		if (!hero) tran.value[id + name + "x"] = x
+		if (!hero) tran.value[id + name + "y"] = y
+
+		const data = {
+			id: id,
+			name: name,
+			x: x,
+			y: y,
+			hero: hero,
+			scalex: scalex,
+			scaley: scaley,
+			start: 0,
+			pause: false,
+			loop: loop,
+			reverse: reverse,
+			pausetime: 0,
+			farme: 0,
+		};
+
+		core.plugin.playing.add(data);
+		return id
+	};
+	this.animateloop = function (id, loop) {
+		core.plugin.playing.forEach(v => {
+			if (!id || v.id === id) v.loop = loop
+		})
+	}
+	this.animatereverse = function (id, reverse) {
+		core.plugin.playing.forEach(v => {
+			if (!id || v.id === id) {
+				const data = flags["animate_" + v.name]
+				if (reverse && !v.reverse) v.start -= (data.allFarme - v.farme * 2) * (1000 / 60)
+				if (!reverse && v.reverse) v.start -= (v.farme * 2 - data.allFarme) * (1000 / 60)
+				v.reverse = reverse
+			}
+		})
+	}
+	this.animatepause = function (id, pause) {
+		core.plugin.playing.forEach(v => {
+			if (!id || v.id === id) v.pause = pause
+		})
+	}
+	this.animateclear = function (id) {
+		core.plugin.playing.forEach(v => {
+			if (!id || v.id === id) core.plugin.playing.delete(v)
+		})
+	}
+	core.registerAnimationFrame("animateonmap", true, function (timestamp) {
+		let frametime = timestamp - thistime
+		thistime = timestamp;
+
+
+		core.clearMap(anctx);
+		core.plugin.playing.forEach((one) => {
+			const data = flags["animate_" + one.name];
+			if (!data) {
+				core.plugin.playing.delete(one);
+			} else {
+				if (one.start === 0) one.start = thistime
+				if (one.pause) one.pausetime += frametime
+				one.farme = Math.floor((thistime - one.start - one.pausetime) / (1000 / 60))
+
+				if (one.reverse) one.farme = data.allFarme - one.farme
+				if ((!one.reverse && one.farme > data.allFarme) || (one.reverse && one.farme < 0)) {
+					if (one.loop) {
+						if (one.reverse) {
+							one.farme = data.allFarme
+							one.start = thistime
+							one.pausetime = 0
+						} else {
+							one.start = thistime
+							one.farme = 0
+							one.pausetime = 0
+						}
+					}
+				}
+				if ((!one.reverse && one.farme > data.allFarme) || (one.reverse && one.farme < 0)) {
+					delete tran.value[one.id + one.name + "x"]
+					delete tran.value[one.id + one.name + "y"]
+					core.plugin.playing.delete(one)
+
+					return
+				}
+
+				data.imageList.forEach(function (image) {
+					if (
+						one.farme >= (image.beforefarme ?? 0) &&
+						one.farme <= (image.afterfarme ?? data.allFarme)
+					) {
+
+						const img = core.material.images.images?.[image.image];
+						if (img) {
+							const gla = image.globalAlpha ?? 100;
+							const agla = image.aglobalAlpha ?? gla,
+								beforefarme = image.beforefarme ?? 0;
+							const afterfarme = image.afterfarme ?? data.allFarme;
+
+							anctx.globalAlpha =
+								(gla +
+									((agla - gla) * (one.farme - beforefarme)) /
+									(afterfarme - beforefarme || 1)) /
+								100;
+
+							const cx =
+								(image.cx ?? 0) +
+								(((image.acx ?? 0) - (image.cx ?? 0)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								cy =
+								(image.cy ?? 0) +
+								(((image.acy ?? 0) - (image.cy ?? 0)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								cw =
+								(image.cw ?? img.width) +
+								(((image.acw ?? img.width) - (image.cw ?? img.width)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								ch =
+								(image.ch ?? img.height) +
+								(((image.acw ?? img.height) - (image.cw ?? img.height)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								x =
+								(image.x ?? 0) +
+								(((image.ax ?? 0) - (image.x ?? 0)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								y =
+								(image.y ?? 0) +
+								(((image.ay ?? 0) - (image.y ?? 0)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								w =
+								(image.w ?? img.width) +
+								(((image.aw ?? img.width) - (image.w ?? img.width)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								h =
+								(image.h ?? img.height) +
+								(((image.aw ?? img.height) - (image.w ?? img.height)) *
+									(one.farme - beforefarme)) /
+								(afterfarme - beforefarme || 1),
+								angle =
+								(Math.PI *
+									((image.angle ?? 0) +
+										(((image.aangle ?? 0) - (image.angle ?? 0)) *
+											(one.farme - beforefarme)) /
+										(afterfarme - beforefarme || 1))) /
+								180;
+
+							if (one.hero) {
+								let sx, sy;
+								if (core.status.heroMoving < 0) {
+									sx = 0;
+									sy = 0;
+								} else {
+									sx =
+										core.utils.scan[core.status.hero.loc.direction].x *
+										4 *
+										core.status.heroMoving;
+									sy =
+										core.utils.scan[core.status.hero.loc.direction].y *
+										4 *
+										core.status.heroMoving;
+								}
+								const herox = core.status.hero.loc.x * 32 + 16 + sx;
+								const heroy = core.status.hero.loc.y * 32 + 16 + sy;
+								core.drawImage(
+									anctx,
+									img,
+									cx,
+									cy,
+									cw,
+									ch,
+									herox + (x - data.px) * one.scalex,
+									heroy + (y - data.py) * one.scaley,
+									w * one.scalex,
+									h * one.scaley,
+									angle
+								);
+
+
+							} else {
+
+								core.drawImage(
+									anctx,
+									img,
+									cx,
+									cy,
+									cw,
+									ch,
+									tran.value[one.id + one.name + "x"] + (x - data.px) * one.scalex,
+									tran.value[one.id + one.name + "y"] + (y - data.py) * one.scaley,
+									w * one.scalex,
+									h * one.scaley,
+									angle
+								);
+							}
+						}
+					}
+				});
+				data.soundList.forEach(function (sound) {
+					const lisen =
+						sound.sound &&
+						core.material.sounds[sound.sound] &&
+						core.musicStatus.soundStatus;
+					if (one.farme == sound.startfarme && lisen) {
+						if (sound.stopbefore) core.stopSound();
+						core.playSound(sound.sound);
+					}
+				});
+
+			}
+		});
+
+	});
+}
 }
\ No newline at end of file
diff --git a/project/sounds/aiy020000190.ogg b/project/sounds/aiy020000190.ogg
new file mode 100644
index 0000000..8212ba6
Binary files /dev/null and b/project/sounds/aiy020000190.ogg differ
diff --git a/project/sounds/aiy020000200.ogg b/project/sounds/aiy020000200.ogg
new file mode 100644
index 0000000..8f710f0
Binary files /dev/null and b/project/sounds/aiy020000200.ogg differ
diff --git a/project/sounds/aiy020000210.ogg b/project/sounds/aiy020000210.ogg
new file mode 100644
index 0000000..e4f719f
Binary files /dev/null and b/project/sounds/aiy020000210.ogg differ
diff --git a/project/sounds/aiy020000220.ogg b/project/sounds/aiy020000220.ogg
new file mode 100644
index 0000000..cf59706
Binary files /dev/null and b/project/sounds/aiy020000220.ogg differ
diff --git a/project/sounds/aiy020000230.ogg b/project/sounds/aiy020000230.ogg
new file mode 100644
index 0000000..c7469e0
Binary files /dev/null and b/project/sounds/aiy020000230.ogg differ
diff --git a/project/sounds/aiy020000240.ogg b/project/sounds/aiy020000240.ogg
new file mode 100644
index 0000000..dc062c2
Binary files /dev/null and b/project/sounds/aiy020000240.ogg differ
diff --git a/project/sounds/aiy020000250.ogg b/project/sounds/aiy020000250.ogg
new file mode 100644
index 0000000..19608c4
Binary files /dev/null and b/project/sounds/aiy020000250.ogg differ
diff --git a/project/sounds/aiy020000260.ogg b/project/sounds/aiy020000260.ogg
new file mode 100644
index 0000000..fc86122
Binary files /dev/null and b/project/sounds/aiy020000260.ogg differ
diff --git a/project/sounds/aiy020000270.ogg b/project/sounds/aiy020000270.ogg
new file mode 100644
index 0000000..8a9e875
Binary files /dev/null and b/project/sounds/aiy020000270.ogg differ
diff --git a/project/sounds/aiy020000280.ogg b/project/sounds/aiy020000280.ogg
new file mode 100644
index 0000000..804c47b
Binary files /dev/null and b/project/sounds/aiy020000280.ogg differ
diff --git a/project/sounds/aiy020000290.ogg b/project/sounds/aiy020000290.ogg
new file mode 100644
index 0000000..46e1502
Binary files /dev/null and b/project/sounds/aiy020000290.ogg differ
diff --git a/project/sounds/aiy020000300.ogg b/project/sounds/aiy020000300.ogg
new file mode 100644
index 0000000..d5445c7
Binary files /dev/null and b/project/sounds/aiy020000300.ogg differ
diff --git a/project/sounds/aiy020000310.ogg b/project/sounds/aiy020000310.ogg
new file mode 100644
index 0000000..e197805
Binary files /dev/null and b/project/sounds/aiy020000310.ogg differ
diff --git a/project/sounds/aiy020000320.ogg b/project/sounds/aiy020000320.ogg
new file mode 100644
index 0000000..eadca46
Binary files /dev/null and b/project/sounds/aiy020000320.ogg differ
diff --git a/project/sounds/aiy020000330.ogg b/project/sounds/aiy020000330.ogg
new file mode 100644
index 0000000..24af673
Binary files /dev/null and b/project/sounds/aiy020000330.ogg differ
diff --git a/project/sounds/aiy020000340.ogg b/project/sounds/aiy020000340.ogg
new file mode 100644
index 0000000..34d7e41
Binary files /dev/null and b/project/sounds/aiy020000340.ogg differ
diff --git a/project/sounds/aiy020000350.ogg b/project/sounds/aiy020000350.ogg
new file mode 100644
index 0000000..25d0572
Binary files /dev/null and b/project/sounds/aiy020000350.ogg differ
diff --git a/project/sounds/aiy020000360.ogg b/project/sounds/aiy020000360.ogg
new file mode 100644
index 0000000..e46bc3e
Binary files /dev/null and b/project/sounds/aiy020000360.ogg differ
diff --git a/project/sounds/aiy020000370.ogg b/project/sounds/aiy020000370.ogg
new file mode 100644
index 0000000..f929ecb
Binary files /dev/null and b/project/sounds/aiy020000370.ogg differ
diff --git a/project/sounds/aiy030000010.ogg b/project/sounds/aiy030000010.ogg
new file mode 100644
index 0000000..9b9d46d
Binary files /dev/null and b/project/sounds/aiy030000010.ogg differ
diff --git a/project/sounds/aiy030000020.ogg b/project/sounds/aiy030000020.ogg
new file mode 100644
index 0000000..a0f4ebb
Binary files /dev/null and b/project/sounds/aiy030000020.ogg differ
diff --git a/project/sounds/aiy030000030.ogg b/project/sounds/aiy030000030.ogg
new file mode 100644
index 0000000..14a2ca8
Binary files /dev/null and b/project/sounds/aiy030000030.ogg differ
diff --git a/project/sounds/aiy030000050.ogg b/project/sounds/aiy030000050.ogg
new file mode 100644
index 0000000..9bc3077
Binary files /dev/null and b/project/sounds/aiy030000050.ogg differ
diff --git a/project/sounds/aiy030000060.ogg b/project/sounds/aiy030000060.ogg
new file mode 100644
index 0000000..ff055c2
Binary files /dev/null and b/project/sounds/aiy030000060.ogg differ
diff --git a/project/sounds/aiy030000070.ogg b/project/sounds/aiy030000070.ogg
new file mode 100644
index 0000000..2e7dcba
Binary files /dev/null and b/project/sounds/aiy030000070.ogg differ
diff --git a/project/sounds/aiy030000080.ogg b/project/sounds/aiy030000080.ogg
new file mode 100644
index 0000000..efc3d38
Binary files /dev/null and b/project/sounds/aiy030000080.ogg differ
diff --git a/project/sounds/aiy030000090.ogg b/project/sounds/aiy030000090.ogg
new file mode 100644
index 0000000..8f9fb91
Binary files /dev/null and b/project/sounds/aiy030000090.ogg differ
diff --git a/project/sounds/aiy030000100.ogg b/project/sounds/aiy030000100.ogg
new file mode 100644
index 0000000..0dbf5bc
Binary files /dev/null and b/project/sounds/aiy030000100.ogg differ
diff --git a/project/sounds/aiy030000110.ogg b/project/sounds/aiy030000110.ogg
new file mode 100644
index 0000000..f722e49
Binary files /dev/null and b/project/sounds/aiy030000110.ogg differ
diff --git a/project/sounds/aiy030000120.ogg b/project/sounds/aiy030000120.ogg
new file mode 100644
index 0000000..53ac3bf
Binary files /dev/null and b/project/sounds/aiy030000120.ogg differ
diff --git a/project/sounds/aiy030000130.ogg b/project/sounds/aiy030000130.ogg
new file mode 100644
index 0000000..aa157af
Binary files /dev/null and b/project/sounds/aiy030000130.ogg differ
diff --git a/project/sounds/aiy050000010.ogg b/project/sounds/aiy050000010.ogg
new file mode 100644
index 0000000..6a9a4a2
Binary files /dev/null and b/project/sounds/aiy050000010.ogg differ
diff --git a/project/sounds/aiy050000020.ogg b/project/sounds/aiy050000020.ogg
new file mode 100644
index 0000000..9b9cdf8
Binary files /dev/null and b/project/sounds/aiy050000020.ogg differ
diff --git a/project/sounds/aiy050000030.ogg b/project/sounds/aiy050000030.ogg
new file mode 100644
index 0000000..d328ca8
Binary files /dev/null and b/project/sounds/aiy050000030.ogg differ
diff --git a/project/sounds/aiy050000040.ogg b/project/sounds/aiy050000040.ogg
new file mode 100644
index 0000000..cc11d60
Binary files /dev/null and b/project/sounds/aiy050000040.ogg differ
diff --git a/project/sounds/aiy050000050.ogg b/project/sounds/aiy050000050.ogg
new file mode 100644
index 0000000..590349f
Binary files /dev/null and b/project/sounds/aiy050000050.ogg differ
diff --git a/project/sounds/aiy050000060.ogg b/project/sounds/aiy050000060.ogg
new file mode 100644
index 0000000..a77d3ba
Binary files /dev/null and b/project/sounds/aiy050000060.ogg differ
diff --git a/project/sounds/aiy050000070.ogg b/project/sounds/aiy050000070.ogg
new file mode 100644
index 0000000..9213193
Binary files /dev/null and b/project/sounds/aiy050000070.ogg differ
diff --git a/project/sounds/aiy050000080.ogg b/project/sounds/aiy050000080.ogg
new file mode 100644
index 0000000..29a2c70
Binary files /dev/null and b/project/sounds/aiy050000080.ogg differ
diff --git a/project/sounds/aiy050000090.ogg b/project/sounds/aiy050000090.ogg
new file mode 100644
index 0000000..b0e5931
Binary files /dev/null and b/project/sounds/aiy050000090.ogg differ
diff --git a/project/sounds/aiy050000100.ogg b/project/sounds/aiy050000100.ogg
new file mode 100644
index 0000000..59e7ebc
Binary files /dev/null and b/project/sounds/aiy050000100.ogg differ
diff --git a/project/sounds/aiy050000110.ogg b/project/sounds/aiy050000110.ogg
new file mode 100644
index 0000000..e198a48
Binary files /dev/null and b/project/sounds/aiy050000110.ogg differ
diff --git a/project/sounds/aiy050000120.ogg b/project/sounds/aiy050000120.ogg
new file mode 100644
index 0000000..ac4bf72
Binary files /dev/null and b/project/sounds/aiy050000120.ogg differ
diff --git a/project/sounds/aiy120000020.ogg b/project/sounds/aiy120000020.ogg
new file mode 100644
index 0000000..10d2ae0
Binary files /dev/null and b/project/sounds/aiy120000020.ogg differ
diff --git a/project/sounds/aiy120000030.ogg b/project/sounds/aiy120000030.ogg
new file mode 100644
index 0000000..2a0dc02
Binary files /dev/null and b/project/sounds/aiy120000030.ogg differ
diff --git a/project/sounds/aiy120000040.ogg b/project/sounds/aiy120000040.ogg
new file mode 100644
index 0000000..f9f0f66
Binary files /dev/null and b/project/sounds/aiy120000040.ogg differ
diff --git a/project/sounds/aiy120000050.ogg b/project/sounds/aiy120000050.ogg
new file mode 100644
index 0000000..0cb03e6
Binary files /dev/null and b/project/sounds/aiy120000050.ogg differ
diff --git a/project/sounds/aiy120000060.ogg b/project/sounds/aiy120000060.ogg
new file mode 100644
index 0000000..bdf482e
Binary files /dev/null and b/project/sounds/aiy120000060.ogg differ
diff --git a/project/sounds/aiy120000070.ogg b/project/sounds/aiy120000070.ogg
new file mode 100644
index 0000000..c8822ac
Binary files /dev/null and b/project/sounds/aiy120000070.ogg differ
diff --git a/project/sounds/aiy120000080.ogg b/project/sounds/aiy120000080.ogg
new file mode 100644
index 0000000..3cd1ff4
Binary files /dev/null and b/project/sounds/aiy120000080.ogg differ
diff --git a/project/sounds/aiy120000090.ogg b/project/sounds/aiy120000090.ogg
new file mode 100644
index 0000000..306c3d4
Binary files /dev/null and b/project/sounds/aiy120000090.ogg differ
diff --git a/project/sounds/aiy120000100.ogg b/project/sounds/aiy120000100.ogg
new file mode 100644
index 0000000..d6597f6
Binary files /dev/null and b/project/sounds/aiy120000100.ogg differ
diff --git a/project/sounds/aiy120000110.ogg b/project/sounds/aiy120000110.ogg
new file mode 100644
index 0000000..a322097
Binary files /dev/null and b/project/sounds/aiy120000110.ogg differ
diff --git a/project/sounds/aiy120000120.ogg b/project/sounds/aiy120000120.ogg
new file mode 100644
index 0000000..acb82fb
Binary files /dev/null and b/project/sounds/aiy120000120.ogg differ
diff --git a/project/sounds/aiy120000130.ogg b/project/sounds/aiy120000130.ogg
new file mode 100644
index 0000000..8f231aa
Binary files /dev/null and b/project/sounds/aiy120000130.ogg differ
diff --git a/project/sounds/aiy120000140.ogg b/project/sounds/aiy120000140.ogg
new file mode 100644
index 0000000..d3bbbb2
Binary files /dev/null and b/project/sounds/aiy120000140.ogg differ
diff --git a/project/sounds/aiy120000150.ogg b/project/sounds/aiy120000150.ogg
new file mode 100644
index 0000000..47f6a04
Binary files /dev/null and b/project/sounds/aiy120000150.ogg differ
diff --git a/project/sounds/aiy120000160.ogg b/project/sounds/aiy120000160.ogg
new file mode 100644
index 0000000..214c63b
Binary files /dev/null and b/project/sounds/aiy120000160.ogg differ
diff --git a/project/sounds/aiy120000170.ogg b/project/sounds/aiy120000170.ogg
new file mode 100644
index 0000000..35d2f1b
Binary files /dev/null and b/project/sounds/aiy120000170.ogg differ
diff --git a/project/sounds/aiy120000180.ogg b/project/sounds/aiy120000180.ogg
new file mode 100644
index 0000000..3417af8
Binary files /dev/null and b/project/sounds/aiy120000180.ogg differ
diff --git a/project/sounds/aiy120000190.ogg b/project/sounds/aiy120000190.ogg
new file mode 100644
index 0000000..37a6a52
Binary files /dev/null and b/project/sounds/aiy120000190.ogg differ
diff --git a/project/sounds/aiy120000200.ogg b/project/sounds/aiy120000200.ogg
new file mode 100644
index 0000000..18eefb3
Binary files /dev/null and b/project/sounds/aiy120000200.ogg differ
diff --git a/project/sounds/aiy120000210.ogg b/project/sounds/aiy120000210.ogg
new file mode 100644
index 0000000..67784d0
Binary files /dev/null and b/project/sounds/aiy120000210.ogg differ
diff --git a/project/sounds/aiy120000220.ogg b/project/sounds/aiy120000220.ogg
new file mode 100644
index 0000000..1e9d440
Binary files /dev/null and b/project/sounds/aiy120000220.ogg differ
diff --git a/project/sounds/aiy120000230.ogg b/project/sounds/aiy120000230.ogg
new file mode 100644
index 0000000..dc4d494
Binary files /dev/null and b/project/sounds/aiy120000230.ogg differ
diff --git a/project/sounds/aiy120000240.ogg b/project/sounds/aiy120000240.ogg
new file mode 100644
index 0000000..88a9404
Binary files /dev/null and b/project/sounds/aiy120000240.ogg differ
diff --git a/project/sounds/aiy120000250.ogg b/project/sounds/aiy120000250.ogg
new file mode 100644
index 0000000..44e9970
Binary files /dev/null and b/project/sounds/aiy120000250.ogg differ
diff --git a/project/sounds/aiy120000260.ogg b/project/sounds/aiy120000260.ogg
new file mode 100644
index 0000000..c93d37b
Binary files /dev/null and b/project/sounds/aiy120000260.ogg differ
diff --git a/project/sounds/aiy120000270.ogg b/project/sounds/aiy120000270.ogg
new file mode 100644
index 0000000..97235e0
Binary files /dev/null and b/project/sounds/aiy120000270.ogg differ
diff --git a/project/sounds/aiy120000280.ogg b/project/sounds/aiy120000280.ogg
new file mode 100644
index 0000000..7153660
Binary files /dev/null and b/project/sounds/aiy120000280.ogg differ
diff --git a/project/sounds/aiy120000290.ogg b/project/sounds/aiy120000290.ogg
new file mode 100644
index 0000000..8c4db5b
Binary files /dev/null and b/project/sounds/aiy120000290.ogg differ
diff --git a/project/sounds/aiy120000300.ogg b/project/sounds/aiy120000300.ogg
new file mode 100644
index 0000000..875d1f7
Binary files /dev/null and b/project/sounds/aiy120000300.ogg differ
diff --git a/project/sounds/aiy120000310.ogg b/project/sounds/aiy120000310.ogg
new file mode 100644
index 0000000..8720a02
Binary files /dev/null and b/project/sounds/aiy120000310.ogg differ
diff --git a/project/sounds/aiy120000320.ogg b/project/sounds/aiy120000320.ogg
new file mode 100644
index 0000000..c86806b
Binary files /dev/null and b/project/sounds/aiy120000320.ogg differ
diff --git a/project/sounds/aiy120000330.ogg b/project/sounds/aiy120000330.ogg
new file mode 100644
index 0000000..0f21adc
Binary files /dev/null and b/project/sounds/aiy120000330.ogg differ
diff --git a/project/sounds/aiy120000340.ogg b/project/sounds/aiy120000340.ogg
new file mode 100644
index 0000000..a055a78
Binary files /dev/null and b/project/sounds/aiy120000340.ogg differ
diff --git a/project/sounds/aiy120000350.ogg b/project/sounds/aiy120000350.ogg
new file mode 100644
index 0000000..0711a4f
Binary files /dev/null and b/project/sounds/aiy120000350.ogg differ
diff --git a/project/sounds/aiy120000360.ogg b/project/sounds/aiy120000360.ogg
new file mode 100644
index 0000000..358c450
Binary files /dev/null and b/project/sounds/aiy120000360.ogg differ
diff --git a/project/sounds/aiy120000370.ogg b/project/sounds/aiy120000370.ogg
new file mode 100644
index 0000000..6381050
Binary files /dev/null and b/project/sounds/aiy120000370.ogg differ
diff --git a/project/sounds/aiy120000380.ogg b/project/sounds/aiy120000380.ogg
new file mode 100644
index 0000000..b80c4f0
Binary files /dev/null and b/project/sounds/aiy120000380.ogg differ
diff --git a/project/sounds/aiy120000390.ogg b/project/sounds/aiy120000390.ogg
new file mode 100644
index 0000000..c608bba
Binary files /dev/null and b/project/sounds/aiy120000390.ogg differ
diff --git a/project/sounds/aiy120000400.ogg b/project/sounds/aiy120000400.ogg
new file mode 100644
index 0000000..656ae27
Binary files /dev/null and b/project/sounds/aiy120000400.ogg differ
diff --git a/project/sounds/aiy120000410.ogg b/project/sounds/aiy120000410.ogg
new file mode 100644
index 0000000..5f04e20
Binary files /dev/null and b/project/sounds/aiy120000410.ogg differ
diff --git a/project/sounds/aiy120000420.ogg b/project/sounds/aiy120000420.ogg
new file mode 100644
index 0000000..86e91fa
Binary files /dev/null and b/project/sounds/aiy120000420.ogg differ
diff --git a/project/sounds/aiy120000430.ogg b/project/sounds/aiy120000430.ogg
new file mode 100644
index 0000000..957e986
Binary files /dev/null and b/project/sounds/aiy120000430.ogg differ
diff --git a/project/sounds/aiy120000440.ogg b/project/sounds/aiy120000440.ogg
new file mode 100644
index 0000000..45231db
Binary files /dev/null and b/project/sounds/aiy120000440.ogg differ
diff --git a/project/sounds/aiy120000450.ogg b/project/sounds/aiy120000450.ogg
new file mode 100644
index 0000000..ad04858
Binary files /dev/null and b/project/sounds/aiy120000450.ogg differ
diff --git a/project/sounds/aiy120000460.ogg b/project/sounds/aiy120000460.ogg
new file mode 100644
index 0000000..d35ded8
Binary files /dev/null and b/project/sounds/aiy120000460.ogg differ
diff --git a/project/sounds/aiy120000470.ogg b/project/sounds/aiy120000470.ogg
new file mode 100644
index 0000000..546a039
Binary files /dev/null and b/project/sounds/aiy120000470.ogg differ
diff --git a/project/sounds/aiy120000480.ogg b/project/sounds/aiy120000480.ogg
new file mode 100644
index 0000000..65f673a
Binary files /dev/null and b/project/sounds/aiy120000480.ogg differ
diff --git a/project/sounds/aiy120000490.ogg b/project/sounds/aiy120000490.ogg
new file mode 100644
index 0000000..36a5df2
Binary files /dev/null and b/project/sounds/aiy120000490.ogg differ
diff --git a/project/sounds/aiy120000500.ogg b/project/sounds/aiy120000500.ogg
new file mode 100644
index 0000000..db40172
Binary files /dev/null and b/project/sounds/aiy120000500.ogg differ
diff --git a/project/sounds/aiy120000510.ogg b/project/sounds/aiy120000510.ogg
new file mode 100644
index 0000000..0da2bbd
Binary files /dev/null and b/project/sounds/aiy120000510.ogg differ
diff --git a/project/sounds/aiy120000520.ogg b/project/sounds/aiy120000520.ogg
new file mode 100644
index 0000000..30618ea
Binary files /dev/null and b/project/sounds/aiy120000520.ogg differ
diff --git a/project/sounds/aiy120000530.ogg b/project/sounds/aiy120000530.ogg
new file mode 100644
index 0000000..63424e5
Binary files /dev/null and b/project/sounds/aiy120000530.ogg differ
diff --git a/project/sounds/aiy120000540.ogg b/project/sounds/aiy120000540.ogg
new file mode 100644
index 0000000..a491c40
Binary files /dev/null and b/project/sounds/aiy120000540.ogg differ
diff --git a/project/sounds/aiy120000550.ogg b/project/sounds/aiy120000550.ogg
new file mode 100644
index 0000000..1adb571
Binary files /dev/null and b/project/sounds/aiy120000550.ogg differ
diff --git a/project/sounds/aiy120000560.ogg b/project/sounds/aiy120000560.ogg
new file mode 100644
index 0000000..eafe149
Binary files /dev/null and b/project/sounds/aiy120000560.ogg differ
diff --git a/project/sounds/aiy120000570.ogg b/project/sounds/aiy120000570.ogg
new file mode 100644
index 0000000..f25d9d2
Binary files /dev/null and b/project/sounds/aiy120000570.ogg differ
diff --git a/project/sounds/aiy120000580.ogg b/project/sounds/aiy120000580.ogg
new file mode 100644
index 0000000..503d4e0
Binary files /dev/null and b/project/sounds/aiy120000580.ogg differ
diff --git a/project/sounds/aiy120000590.ogg b/project/sounds/aiy120000590.ogg
new file mode 100644
index 0000000..0fa071e
Binary files /dev/null and b/project/sounds/aiy120000590.ogg differ
diff --git a/project/sounds/aiy120000600.ogg b/project/sounds/aiy120000600.ogg
new file mode 100644
index 0000000..6d22156
Binary files /dev/null and b/project/sounds/aiy120000600.ogg differ
diff --git a/project/sounds/aiy120000610.ogg b/project/sounds/aiy120000610.ogg
new file mode 100644
index 0000000..95b2a7d
Binary files /dev/null and b/project/sounds/aiy120000610.ogg differ
diff --git a/project/sounds/aiy120000620.ogg b/project/sounds/aiy120000620.ogg
new file mode 100644
index 0000000..f03cf8f
Binary files /dev/null and b/project/sounds/aiy120000620.ogg differ
diff --git a/project/sounds/aiy120000630.ogg b/project/sounds/aiy120000630.ogg
new file mode 100644
index 0000000..5420074
Binary files /dev/null and b/project/sounds/aiy120000630.ogg differ
diff --git a/project/sounds/aiy120000640.ogg b/project/sounds/aiy120000640.ogg
new file mode 100644
index 0000000..a204879
Binary files /dev/null and b/project/sounds/aiy120000640.ogg differ
diff --git a/project/sounds/aiy120000650.ogg b/project/sounds/aiy120000650.ogg
new file mode 100644
index 0000000..257b817
Binary files /dev/null and b/project/sounds/aiy120000650.ogg differ
diff --git a/project/sounds/aiy120000670.ogg b/project/sounds/aiy120000670.ogg
new file mode 100644
index 0000000..b603efe
Binary files /dev/null and b/project/sounds/aiy120000670.ogg differ
diff --git a/project/sounds/aiy120000680.ogg b/project/sounds/aiy120000680.ogg
new file mode 100644
index 0000000..32f2bdd
Binary files /dev/null and b/project/sounds/aiy120000680.ogg differ
diff --git a/project/sounds/aiy120000690.ogg b/project/sounds/aiy120000690.ogg
new file mode 100644
index 0000000..89a831c
Binary files /dev/null and b/project/sounds/aiy120000690.ogg differ
diff --git a/project/sounds/aiy130000010.ogg b/project/sounds/aiy130000010.ogg
new file mode 100644
index 0000000..fe0e4e2
Binary files /dev/null and b/project/sounds/aiy130000010.ogg differ
diff --git a/project/sounds/aiy130000020.ogg b/project/sounds/aiy130000020.ogg
new file mode 100644
index 0000000..6e13850
Binary files /dev/null and b/project/sounds/aiy130000020.ogg differ
diff --git a/project/sounds/aiy150000010.ogg b/project/sounds/aiy150000010.ogg
new file mode 100644
index 0000000..0659452
Binary files /dev/null and b/project/sounds/aiy150000010.ogg differ
diff --git a/project/sounds/aiy150000020.ogg b/project/sounds/aiy150000020.ogg
new file mode 100644
index 0000000..798a9fc
Binary files /dev/null and b/project/sounds/aiy150000020.ogg differ
diff --git a/project/sounds/aiy150000030.ogg b/project/sounds/aiy150000030.ogg
new file mode 100644
index 0000000..9022052
Binary files /dev/null and b/project/sounds/aiy150000030.ogg differ
diff --git a/project/sounds/aiy150000040.ogg b/project/sounds/aiy150000040.ogg
new file mode 100644
index 0000000..c6e6233
Binary files /dev/null and b/project/sounds/aiy150000040.ogg differ
diff --git a/project/sounds/aiy150000050.ogg b/project/sounds/aiy150000050.ogg
new file mode 100644
index 0000000..8764aee
Binary files /dev/null and b/project/sounds/aiy150000050.ogg differ
diff --git a/project/sounds/aiy150000060.ogg b/project/sounds/aiy150000060.ogg
new file mode 100644
index 0000000..e305004
Binary files /dev/null and b/project/sounds/aiy150000060.ogg differ
diff --git a/project/sounds/aiy150000070.ogg b/project/sounds/aiy150000070.ogg
new file mode 100644
index 0000000..e2d173a
Binary files /dev/null and b/project/sounds/aiy150000070.ogg differ
diff --git a/project/sounds/aiy150000080.ogg b/project/sounds/aiy150000080.ogg
new file mode 100644
index 0000000..8bed0cf
Binary files /dev/null and b/project/sounds/aiy150000080.ogg differ
diff --git a/project/sounds/aiy150000090.ogg b/project/sounds/aiy150000090.ogg
new file mode 100644
index 0000000..8c35672
Binary files /dev/null and b/project/sounds/aiy150000090.ogg differ
diff --git a/project/sounds/aiy150000100.ogg b/project/sounds/aiy150000100.ogg
new file mode 100644
index 0000000..0fda997
Binary files /dev/null and b/project/sounds/aiy150000100.ogg differ
diff --git a/project/sounds/aiy150000110.ogg b/project/sounds/aiy150000110.ogg
new file mode 100644
index 0000000..2e7da86
Binary files /dev/null and b/project/sounds/aiy150000110.ogg differ
diff --git a/project/sounds/aiy150000120.ogg b/project/sounds/aiy150000120.ogg
new file mode 100644
index 0000000..1e6adc7
Binary files /dev/null and b/project/sounds/aiy150000120.ogg differ
diff --git a/project/sounds/aiy150000130.ogg b/project/sounds/aiy150000130.ogg
new file mode 100644
index 0000000..bdbefb9
Binary files /dev/null and b/project/sounds/aiy150000130.ogg differ
diff --git a/project/sounds/aiy150000140.ogg b/project/sounds/aiy150000140.ogg
new file mode 100644
index 0000000..5ec6eda
Binary files /dev/null and b/project/sounds/aiy150000140.ogg differ
diff --git a/project/sounds/aiy150000150.ogg b/project/sounds/aiy150000150.ogg
new file mode 100644
index 0000000..28fad01
Binary files /dev/null and b/project/sounds/aiy150000150.ogg differ
diff --git a/project/sounds/aiy150000160.ogg b/project/sounds/aiy150000160.ogg
new file mode 100644
index 0000000..54bb342
Binary files /dev/null and b/project/sounds/aiy150000160.ogg differ
diff --git a/project/sounds/aiy150000170.ogg b/project/sounds/aiy150000170.ogg
new file mode 100644
index 0000000..44543c4
Binary files /dev/null and b/project/sounds/aiy150000170.ogg differ
diff --git a/project/sounds/aiy150000180.ogg b/project/sounds/aiy150000180.ogg
new file mode 100644
index 0000000..16682cc
Binary files /dev/null and b/project/sounds/aiy150000180.ogg differ
diff --git a/project/sounds/aiy150000190.ogg b/project/sounds/aiy150000190.ogg
new file mode 100644
index 0000000..74b719a
Binary files /dev/null and b/project/sounds/aiy150000190.ogg differ
diff --git a/project/sounds/aiy150000200.ogg b/project/sounds/aiy150000200.ogg
new file mode 100644
index 0000000..c6c73df
Binary files /dev/null and b/project/sounds/aiy150000200.ogg differ
diff --git a/project/sounds/aiy150000210.ogg b/project/sounds/aiy150000210.ogg
new file mode 100644
index 0000000..bb1182b
Binary files /dev/null and b/project/sounds/aiy150000210.ogg differ
diff --git a/project/sounds/aiy150000220.ogg b/project/sounds/aiy150000220.ogg
new file mode 100644
index 0000000..3babde5
Binary files /dev/null and b/project/sounds/aiy150000220.ogg differ
diff --git a/project/sounds/aiy150000230.ogg b/project/sounds/aiy150000230.ogg
new file mode 100644
index 0000000..a51169e
Binary files /dev/null and b/project/sounds/aiy150000230.ogg differ
diff --git a/project/sounds/aiy150000240.ogg b/project/sounds/aiy150000240.ogg
new file mode 100644
index 0000000..6ff7544
Binary files /dev/null and b/project/sounds/aiy150000240.ogg differ
diff --git a/project/sounds/aiy150000250.ogg b/project/sounds/aiy150000250.ogg
new file mode 100644
index 0000000..1b3acf8
Binary files /dev/null and b/project/sounds/aiy150000250.ogg differ
diff --git a/project/sounds/aiy310000310.ogg b/project/sounds/aiy310000310.ogg
new file mode 100644
index 0000000..7d1eda9
Binary files /dev/null and b/project/sounds/aiy310000310.ogg differ
diff --git a/project/sounds/aiy310000320.ogg b/project/sounds/aiy310000320.ogg
new file mode 100644
index 0000000..2793e41
Binary files /dev/null and b/project/sounds/aiy310000320.ogg differ
diff --git a/project/sounds/aiy310000330.ogg b/project/sounds/aiy310000330.ogg
new file mode 100644
index 0000000..f3806c3
Binary files /dev/null and b/project/sounds/aiy310000330.ogg differ
diff --git a/project/sounds/aiy310000340.ogg b/project/sounds/aiy310000340.ogg
new file mode 100644
index 0000000..a640644
Binary files /dev/null and b/project/sounds/aiy310000340.ogg differ
diff --git a/project/sounds/aiy310000350.ogg b/project/sounds/aiy310000350.ogg
new file mode 100644
index 0000000..ea5352e
Binary files /dev/null and b/project/sounds/aiy310000350.ogg differ
diff --git a/project/sounds/aiy310000360.ogg b/project/sounds/aiy310000360.ogg
new file mode 100644
index 0000000..5eefa11
Binary files /dev/null and b/project/sounds/aiy310000360.ogg differ
diff --git a/project/sounds/aiy310000370.ogg b/project/sounds/aiy310000370.ogg
new file mode 100644
index 0000000..91a7a46
Binary files /dev/null and b/project/sounds/aiy310000370.ogg differ
diff --git a/project/sounds/aiy310000380.ogg b/project/sounds/aiy310000380.ogg
new file mode 100644
index 0000000..7ba0311
Binary files /dev/null and b/project/sounds/aiy310000380.ogg differ
diff --git a/project/sounds/aiy310000390.ogg b/project/sounds/aiy310000390.ogg
new file mode 100644
index 0000000..3c877e9
Binary files /dev/null and b/project/sounds/aiy310000390.ogg differ
diff --git a/project/sounds/aiy310000400.ogg b/project/sounds/aiy310000400.ogg
new file mode 100644
index 0000000..df3e50c
Binary files /dev/null and b/project/sounds/aiy310000400.ogg differ
diff --git a/project/sounds/aiy310000410.ogg b/project/sounds/aiy310000410.ogg
new file mode 100644
index 0000000..69ad557
Binary files /dev/null and b/project/sounds/aiy310000410.ogg differ
diff --git a/project/sounds/aiy310000420.ogg b/project/sounds/aiy310000420.ogg
new file mode 100644
index 0000000..1a50529
Binary files /dev/null and b/project/sounds/aiy310000420.ogg differ
diff --git a/project/sounds/aiy310000430.ogg b/project/sounds/aiy310000430.ogg
new file mode 100644
index 0000000..980b7af
Binary files /dev/null and b/project/sounds/aiy310000430.ogg differ
diff --git a/project/sounds/aiy310000440.ogg b/project/sounds/aiy310000440.ogg
new file mode 100644
index 0000000..68e063a
Binary files /dev/null and b/project/sounds/aiy310000440.ogg differ
diff --git a/project/sounds/aiy310000450.ogg b/project/sounds/aiy310000450.ogg
new file mode 100644
index 0000000..991e23c
Binary files /dev/null and b/project/sounds/aiy310000450.ogg differ
diff --git a/project/sounds/aiy310000460.ogg b/project/sounds/aiy310000460.ogg
new file mode 100644
index 0000000..7208098
Binary files /dev/null and b/project/sounds/aiy310000460.ogg differ
diff --git a/project/sounds/aiy310000470.ogg b/project/sounds/aiy310000470.ogg
new file mode 100644
index 0000000..fe71d8f
Binary files /dev/null and b/project/sounds/aiy310000470.ogg differ
diff --git a/project/sounds/aiy310000480.ogg b/project/sounds/aiy310000480.ogg
new file mode 100644
index 0000000..11bc4fe
Binary files /dev/null and b/project/sounds/aiy310000480.ogg differ
diff --git a/project/sounds/aiy310000490.ogg b/project/sounds/aiy310000490.ogg
new file mode 100644
index 0000000..4a8c1a5
Binary files /dev/null and b/project/sounds/aiy310000490.ogg differ
diff --git a/project/sounds/aiy310000510.ogg b/project/sounds/aiy310000510.ogg
new file mode 100644
index 0000000..5f98abb
Binary files /dev/null and b/project/sounds/aiy310000510.ogg differ
diff --git a/project/sounds/aiy310000520.ogg b/project/sounds/aiy310000520.ogg
new file mode 100644
index 0000000..5dbe058
Binary files /dev/null and b/project/sounds/aiy310000520.ogg differ
diff --git a/project/sounds/aiy310000530.ogg b/project/sounds/aiy310000530.ogg
new file mode 100644
index 0000000..317a75e
Binary files /dev/null and b/project/sounds/aiy310000530.ogg differ
diff --git a/project/sounds/aiy310000540.ogg b/project/sounds/aiy310000540.ogg
new file mode 100644
index 0000000..ebdceab
Binary files /dev/null and b/project/sounds/aiy310000540.ogg differ
diff --git a/project/sounds/aiy310000550.ogg b/project/sounds/aiy310000550.ogg
new file mode 100644
index 0000000..735881c
Binary files /dev/null and b/project/sounds/aiy310000550.ogg differ
diff --git a/project/sounds/aiy310000560.ogg b/project/sounds/aiy310000560.ogg
new file mode 100644
index 0000000..a631836
Binary files /dev/null and b/project/sounds/aiy310000560.ogg differ
diff --git a/project/sounds/aiy310000570.ogg b/project/sounds/aiy310000570.ogg
new file mode 100644
index 0000000..cd365e2
Binary files /dev/null and b/project/sounds/aiy310000570.ogg differ
diff --git a/project/sounds/aiy310000580.ogg b/project/sounds/aiy310000580.ogg
new file mode 100644
index 0000000..1ff4d98
Binary files /dev/null and b/project/sounds/aiy310000580.ogg differ
diff --git a/project/sounds/aiy310000590.ogg b/project/sounds/aiy310000590.ogg
new file mode 100644
index 0000000..d6d161a
Binary files /dev/null and b/project/sounds/aiy310000590.ogg differ
diff --git a/project/sounds/aiy310000600.ogg b/project/sounds/aiy310000600.ogg
new file mode 100644
index 0000000..3afd47d
Binary files /dev/null and b/project/sounds/aiy310000600.ogg differ
diff --git a/project/sounds/aiy310000610.ogg b/project/sounds/aiy310000610.ogg
new file mode 100644
index 0000000..fdcb28b
Binary files /dev/null and b/project/sounds/aiy310000610.ogg differ
diff --git a/project/sounds/aiy310000620.ogg b/project/sounds/aiy310000620.ogg
new file mode 100644
index 0000000..b9d5791
Binary files /dev/null and b/project/sounds/aiy310000620.ogg differ
diff --git a/project/sounds/aiy310000630.ogg b/project/sounds/aiy310000630.ogg
new file mode 100644
index 0000000..b564e34
Binary files /dev/null and b/project/sounds/aiy310000630.ogg differ
diff --git a/project/sounds/aiy310000640.ogg b/project/sounds/aiy310000640.ogg
new file mode 100644
index 0000000..d976e60
Binary files /dev/null and b/project/sounds/aiy310000640.ogg differ
diff --git a/project/sounds/aiy310000650.ogg b/project/sounds/aiy310000650.ogg
new file mode 100644
index 0000000..ea7b117
Binary files /dev/null and b/project/sounds/aiy310000650.ogg differ
diff --git a/project/sounds/aiy310000660.ogg b/project/sounds/aiy310000660.ogg
new file mode 100644
index 0000000..95f02d0
Binary files /dev/null and b/project/sounds/aiy310000660.ogg differ
diff --git a/project/sounds/aiy310000670.ogg b/project/sounds/aiy310000670.ogg
new file mode 100644
index 0000000..17df42a
Binary files /dev/null and b/project/sounds/aiy310000670.ogg differ
diff --git a/project/sounds/aiy310000680.ogg b/project/sounds/aiy310000680.ogg
new file mode 100644
index 0000000..032a9b8
Binary files /dev/null and b/project/sounds/aiy310000680.ogg differ
diff --git a/project/sounds/aiy310000690.ogg b/project/sounds/aiy310000690.ogg
new file mode 100644
index 0000000..9d99453
Binary files /dev/null and b/project/sounds/aiy310000690.ogg differ
diff --git a/project/sounds/aiy310000700.ogg b/project/sounds/aiy310000700.ogg
new file mode 100644
index 0000000..7cefbff
Binary files /dev/null and b/project/sounds/aiy310000700.ogg differ
diff --git a/project/sounds/aiy310000710.ogg b/project/sounds/aiy310000710.ogg
new file mode 100644
index 0000000..dc5eb5a
Binary files /dev/null and b/project/sounds/aiy310000710.ogg differ
diff --git a/project/sounds/aiy310000720.ogg b/project/sounds/aiy310000720.ogg
new file mode 100644
index 0000000..fd37fd3
Binary files /dev/null and b/project/sounds/aiy310000720.ogg differ
diff --git a/project/sounds/aiy310000730.ogg b/project/sounds/aiy310000730.ogg
new file mode 100644
index 0000000..94a4631
Binary files /dev/null and b/project/sounds/aiy310000730.ogg differ
diff --git a/project/sounds/aiy310000740.ogg b/project/sounds/aiy310000740.ogg
new file mode 100644
index 0000000..db5f537
Binary files /dev/null and b/project/sounds/aiy310000740.ogg differ
diff --git a/project/sounds/aiy310000750.ogg b/project/sounds/aiy310000750.ogg
new file mode 100644
index 0000000..90573c8
Binary files /dev/null and b/project/sounds/aiy310000750.ogg differ
diff --git a/project/sounds/aiy310000760.ogg b/project/sounds/aiy310000760.ogg
new file mode 100644
index 0000000..43f2522
Binary files /dev/null and b/project/sounds/aiy310000760.ogg differ
diff --git a/project/sounds/aiy310000770.ogg b/project/sounds/aiy310000770.ogg
new file mode 100644
index 0000000..fccf1f1
Binary files /dev/null and b/project/sounds/aiy310000770.ogg differ
diff --git a/project/sounds/aiy310000780.ogg b/project/sounds/aiy310000780.ogg
new file mode 100644
index 0000000..d1d04ee
Binary files /dev/null and b/project/sounds/aiy310000780.ogg differ
diff --git a/project/sounds/aiy310000790.ogg b/project/sounds/aiy310000790.ogg
new file mode 100644
index 0000000..2cfa70b
Binary files /dev/null and b/project/sounds/aiy310000790.ogg differ
diff --git a/project/sounds/aiy310000800.ogg b/project/sounds/aiy310000800.ogg
new file mode 100644
index 0000000..4400a66
Binary files /dev/null and b/project/sounds/aiy310000800.ogg differ
diff --git a/project/sounds/aiy310000810.ogg b/project/sounds/aiy310000810.ogg
new file mode 100644
index 0000000..4066404
Binary files /dev/null and b/project/sounds/aiy310000810.ogg differ
diff --git a/project/sounds/aiy310000820.ogg b/project/sounds/aiy310000820.ogg
new file mode 100644
index 0000000..29c5807
Binary files /dev/null and b/project/sounds/aiy310000820.ogg differ
diff --git a/project/sounds/aiy310000830.ogg b/project/sounds/aiy310000830.ogg
new file mode 100644
index 0000000..84770ef
Binary files /dev/null and b/project/sounds/aiy310000830.ogg differ
diff --git a/project/sounds/aiy310000840.ogg b/project/sounds/aiy310000840.ogg
new file mode 100644
index 0000000..5b44741
Binary files /dev/null and b/project/sounds/aiy310000840.ogg differ
diff --git a/project/sounds/aiy310000850.ogg b/project/sounds/aiy310000850.ogg
new file mode 100644
index 0000000..0900773
Binary files /dev/null and b/project/sounds/aiy310000850.ogg differ
diff --git a/project/sounds/aiy310000860.ogg b/project/sounds/aiy310000860.ogg
new file mode 100644
index 0000000..e31d562
Binary files /dev/null and b/project/sounds/aiy310000860.ogg differ
diff --git a/project/sounds/aiy310000870.ogg b/project/sounds/aiy310000870.ogg
new file mode 100644
index 0000000..5c9f936
Binary files /dev/null and b/project/sounds/aiy310000870.ogg differ
diff --git a/project/sounds/aiy310000880.ogg b/project/sounds/aiy310000880.ogg
new file mode 100644
index 0000000..0938bb7
Binary files /dev/null and b/project/sounds/aiy310000880.ogg differ
diff --git a/project/sounds/aiy310000890.ogg b/project/sounds/aiy310000890.ogg
new file mode 100644
index 0000000..89e43c5
Binary files /dev/null and b/project/sounds/aiy310000890.ogg differ
diff --git a/project/sounds/aiy310000900.ogg b/project/sounds/aiy310000900.ogg
new file mode 100644
index 0000000..e809e52
Binary files /dev/null and b/project/sounds/aiy310000900.ogg differ
diff --git a/project/sounds/aiy310000910.ogg b/project/sounds/aiy310000910.ogg
new file mode 100644
index 0000000..ffb6b5b
Binary files /dev/null and b/project/sounds/aiy310000910.ogg differ
diff --git a/project/sounds/aiy310000920.ogg b/project/sounds/aiy310000920.ogg
new file mode 100644
index 0000000..f470e54
Binary files /dev/null and b/project/sounds/aiy310000920.ogg differ
diff --git a/project/sounds/aiy310000930.ogg b/project/sounds/aiy310000930.ogg
new file mode 100644
index 0000000..494d15f
Binary files /dev/null and b/project/sounds/aiy310000930.ogg differ
diff --git a/project/sounds/aiy310000940.ogg b/project/sounds/aiy310000940.ogg
new file mode 100644
index 0000000..de36a9d
Binary files /dev/null and b/project/sounds/aiy310000940.ogg differ
diff --git a/project/sounds/aiy310000950.ogg b/project/sounds/aiy310000950.ogg
new file mode 100644
index 0000000..9abfea9
Binary files /dev/null and b/project/sounds/aiy310000950.ogg differ
diff --git a/project/sounds/aiy310000960.ogg b/project/sounds/aiy310000960.ogg
new file mode 100644
index 0000000..394ddcb
Binary files /dev/null and b/project/sounds/aiy310000960.ogg differ
diff --git a/project/sounds/aiy310000970.ogg b/project/sounds/aiy310000970.ogg
new file mode 100644
index 0000000..aefe802
Binary files /dev/null and b/project/sounds/aiy310000970.ogg differ
diff --git a/project/sounds/aiy310000980.ogg b/project/sounds/aiy310000980.ogg
new file mode 100644
index 0000000..9c7ea85
Binary files /dev/null and b/project/sounds/aiy310000980.ogg differ
diff --git a/project/sounds/aiy310000990.ogg b/project/sounds/aiy310000990.ogg
new file mode 100644
index 0000000..5808076
Binary files /dev/null and b/project/sounds/aiy310000990.ogg differ
diff --git a/project/sounds/aiy310001000.ogg b/project/sounds/aiy310001000.ogg
new file mode 100644
index 0000000..2dba8fb
Binary files /dev/null and b/project/sounds/aiy310001000.ogg differ
diff --git a/project/sounds/aiy310001010.ogg b/project/sounds/aiy310001010.ogg
new file mode 100644
index 0000000..a647a12
Binary files /dev/null and b/project/sounds/aiy310001010.ogg differ
diff --git a/project/sounds/aiy310001027.ogg b/project/sounds/aiy310001027.ogg
new file mode 100644
index 0000000..9d31c17
Binary files /dev/null and b/project/sounds/aiy310001027.ogg differ
diff --git a/project/sounds/aiy310001030.ogg b/project/sounds/aiy310001030.ogg
new file mode 100644
index 0000000..c0f363e
Binary files /dev/null and b/project/sounds/aiy310001030.ogg differ
diff --git a/project/sounds/aiy310001040.ogg b/project/sounds/aiy310001040.ogg
new file mode 100644
index 0000000..2d36a51
Binary files /dev/null and b/project/sounds/aiy310001040.ogg differ
diff --git a/project/sounds/aiy310001050.ogg b/project/sounds/aiy310001050.ogg
new file mode 100644
index 0000000..7b3d406
Binary files /dev/null and b/project/sounds/aiy310001050.ogg differ
diff --git a/project/sounds/aiy310001060.ogg b/project/sounds/aiy310001060.ogg
new file mode 100644
index 0000000..5a7c2c1
Binary files /dev/null and b/project/sounds/aiy310001060.ogg differ
diff --git a/project/sounds/aiy310001070.ogg b/project/sounds/aiy310001070.ogg
new file mode 100644
index 0000000..4ecc17a
Binary files /dev/null and b/project/sounds/aiy310001070.ogg differ
diff --git a/project/sounds/aiy310001080.ogg b/project/sounds/aiy310001080.ogg
new file mode 100644
index 0000000..d97e943
Binary files /dev/null and b/project/sounds/aiy310001080.ogg differ
diff --git a/project/sounds/aiy310001090.ogg b/project/sounds/aiy310001090.ogg
new file mode 100644
index 0000000..dc60205
Binary files /dev/null and b/project/sounds/aiy310001090.ogg differ
diff --git a/project/sounds/aiy310001100.ogg b/project/sounds/aiy310001100.ogg
new file mode 100644
index 0000000..13d7d7d
Binary files /dev/null and b/project/sounds/aiy310001100.ogg differ
diff --git a/project/sounds/aiy310001110.ogg b/project/sounds/aiy310001110.ogg
new file mode 100644
index 0000000..040c120
Binary files /dev/null and b/project/sounds/aiy310001110.ogg differ
diff --git a/project/sounds/aiy310001120.ogg b/project/sounds/aiy310001120.ogg
new file mode 100644
index 0000000..458e4f7
Binary files /dev/null and b/project/sounds/aiy310001120.ogg differ
diff --git a/project/sounds/aiy310001130.ogg b/project/sounds/aiy310001130.ogg
new file mode 100644
index 0000000..7daea21
Binary files /dev/null and b/project/sounds/aiy310001130.ogg differ
diff --git a/project/sounds/aiy310001140.ogg b/project/sounds/aiy310001140.ogg
new file mode 100644
index 0000000..7d54e19
Binary files /dev/null and b/project/sounds/aiy310001140.ogg differ
diff --git a/project/sounds/aiy310001150.ogg b/project/sounds/aiy310001150.ogg
new file mode 100644
index 0000000..b06bdc9
Binary files /dev/null and b/project/sounds/aiy310001150.ogg differ
diff --git a/project/sounds/aiy310001160.ogg b/project/sounds/aiy310001160.ogg
new file mode 100644
index 0000000..3ec510b
Binary files /dev/null and b/project/sounds/aiy310001160.ogg differ
diff --git a/project/sounds/aiy310001170.ogg b/project/sounds/aiy310001170.ogg
new file mode 100644
index 0000000..d96b1e2
Binary files /dev/null and b/project/sounds/aiy310001170.ogg differ
diff --git a/project/sounds/aiy310001180.ogg b/project/sounds/aiy310001180.ogg
new file mode 100644
index 0000000..88cffb2
Binary files /dev/null and b/project/sounds/aiy310001180.ogg differ
diff --git a/project/sounds/aiy310001190.ogg b/project/sounds/aiy310001190.ogg
new file mode 100644
index 0000000..e35c481
Binary files /dev/null and b/project/sounds/aiy310001190.ogg differ
diff --git a/project/sounds/aiy310001200.ogg b/project/sounds/aiy310001200.ogg
new file mode 100644
index 0000000..9c30d38
Binary files /dev/null and b/project/sounds/aiy310001200.ogg differ
diff --git a/project/sounds/aiy310001210.ogg b/project/sounds/aiy310001210.ogg
new file mode 100644
index 0000000..cfe51c7
Binary files /dev/null and b/project/sounds/aiy310001210.ogg differ
diff --git a/project/sounds/aiy310001220.ogg b/project/sounds/aiy310001220.ogg
new file mode 100644
index 0000000..e7d5e61
Binary files /dev/null and b/project/sounds/aiy310001220.ogg differ
diff --git a/project/sounds/aiy310001230.ogg b/project/sounds/aiy310001230.ogg
new file mode 100644
index 0000000..4aa7064
Binary files /dev/null and b/project/sounds/aiy310001230.ogg differ
diff --git a/project/sounds/aiy310001240.ogg b/project/sounds/aiy310001240.ogg
new file mode 100644
index 0000000..9342811
Binary files /dev/null and b/project/sounds/aiy310001240.ogg differ
diff --git a/project/sounds/aiy310001250.ogg b/project/sounds/aiy310001250.ogg
new file mode 100644
index 0000000..15a6960
Binary files /dev/null and b/project/sounds/aiy310001250.ogg differ
diff --git a/project/sounds/aiy310001260.ogg b/project/sounds/aiy310001260.ogg
new file mode 100644
index 0000000..dab5516
Binary files /dev/null and b/project/sounds/aiy310001260.ogg differ
diff --git a/project/sounds/aiy310001270.ogg b/project/sounds/aiy310001270.ogg
new file mode 100644
index 0000000..d531b49
Binary files /dev/null and b/project/sounds/aiy310001270.ogg differ
diff --git a/project/sounds/aiy310001280.ogg b/project/sounds/aiy310001280.ogg
new file mode 100644
index 0000000..2e749a4
Binary files /dev/null and b/project/sounds/aiy310001280.ogg differ
diff --git a/project/sounds/aiy310001290.ogg b/project/sounds/aiy310001290.ogg
new file mode 100644
index 0000000..c2e81a6
Binary files /dev/null and b/project/sounds/aiy310001290.ogg differ
diff --git a/project/sounds/aiy310001300.ogg b/project/sounds/aiy310001300.ogg
new file mode 100644
index 0000000..904940f
Binary files /dev/null and b/project/sounds/aiy310001300.ogg differ
diff --git a/project/sounds/aiy310001310.ogg b/project/sounds/aiy310001310.ogg
new file mode 100644
index 0000000..a7f25b0
Binary files /dev/null and b/project/sounds/aiy310001310.ogg differ
diff --git a/project/sounds/aiy310001320.ogg b/project/sounds/aiy310001320.ogg
new file mode 100644
index 0000000..1b06256
Binary files /dev/null and b/project/sounds/aiy310001320.ogg differ
diff --git a/project/sounds/aiy310001330.ogg b/project/sounds/aiy310001330.ogg
new file mode 100644
index 0000000..8525810
Binary files /dev/null and b/project/sounds/aiy310001330.ogg differ
diff --git a/project/sounds/aiy310001340.ogg b/project/sounds/aiy310001340.ogg
new file mode 100644
index 0000000..44b81fe
Binary files /dev/null and b/project/sounds/aiy310001340.ogg differ
diff --git a/project/sounds/aiy310001350.ogg b/project/sounds/aiy310001350.ogg
new file mode 100644
index 0000000..2291f96
Binary files /dev/null and b/project/sounds/aiy310001350.ogg differ
diff --git a/project/sounds/aiy310001360.ogg b/project/sounds/aiy310001360.ogg
new file mode 100644
index 0000000..2114e7b
Binary files /dev/null and b/project/sounds/aiy310001360.ogg differ
diff --git a/project/sounds/aiy310001370.ogg b/project/sounds/aiy310001370.ogg
new file mode 100644
index 0000000..d7df9be
Binary files /dev/null and b/project/sounds/aiy310001370.ogg differ
diff --git a/project/sounds/aiy310001380.ogg b/project/sounds/aiy310001380.ogg
new file mode 100644
index 0000000..fac86f0
Binary files /dev/null and b/project/sounds/aiy310001380.ogg differ
diff --git a/project/sounds/aiy310001390.ogg b/project/sounds/aiy310001390.ogg
new file mode 100644
index 0000000..e543b6f
Binary files /dev/null and b/project/sounds/aiy310001390.ogg differ
diff --git a/project/sounds/aiy320000010.ogg b/project/sounds/aiy320000010.ogg
new file mode 100644
index 0000000..54d6db5
Binary files /dev/null and b/project/sounds/aiy320000010.ogg differ
diff --git a/project/sounds/aiy320000020.ogg b/project/sounds/aiy320000020.ogg
new file mode 100644
index 0000000..d98d94e
Binary files /dev/null and b/project/sounds/aiy320000020.ogg differ
diff --git a/project/sounds/aiy320000030.ogg b/project/sounds/aiy320000030.ogg
new file mode 100644
index 0000000..574f903
Binary files /dev/null and b/project/sounds/aiy320000030.ogg differ
diff --git a/project/sounds/aiy320000040.ogg b/project/sounds/aiy320000040.ogg
new file mode 100644
index 0000000..613c27f
Binary files /dev/null and b/project/sounds/aiy320000040.ogg differ
diff --git a/project/sounds/aiy320000050.ogg b/project/sounds/aiy320000050.ogg
new file mode 100644
index 0000000..b35ca98
Binary files /dev/null and b/project/sounds/aiy320000050.ogg differ
diff --git a/project/sounds/aiy320000060.ogg b/project/sounds/aiy320000060.ogg
new file mode 100644
index 0000000..0365def
Binary files /dev/null and b/project/sounds/aiy320000060.ogg differ
diff --git a/project/sounds/aiy320000070.ogg b/project/sounds/aiy320000070.ogg
new file mode 100644
index 0000000..52e3d11
Binary files /dev/null and b/project/sounds/aiy320000070.ogg differ
diff --git a/project/sounds/aiy320000080.ogg b/project/sounds/aiy320000080.ogg
new file mode 100644
index 0000000..9a2afef
Binary files /dev/null and b/project/sounds/aiy320000080.ogg differ
diff --git a/project/sounds/aiy320000090.ogg b/project/sounds/aiy320000090.ogg
new file mode 100644
index 0000000..608f2e1
Binary files /dev/null and b/project/sounds/aiy320000090.ogg differ
diff --git a/project/sounds/aiy320000100.ogg b/project/sounds/aiy320000100.ogg
new file mode 100644
index 0000000..b97a61c
Binary files /dev/null and b/project/sounds/aiy320000100.ogg differ
diff --git a/project/sounds/aiy320000110.ogg b/project/sounds/aiy320000110.ogg
new file mode 100644
index 0000000..f15348f
Binary files /dev/null and b/project/sounds/aiy320000110.ogg differ
diff --git a/project/sounds/aiy320000120.ogg b/project/sounds/aiy320000120.ogg
new file mode 100644
index 0000000..3adda1d
Binary files /dev/null and b/project/sounds/aiy320000120.ogg differ
diff --git a/project/sounds/aiy320000135.ogg b/project/sounds/aiy320000135.ogg
new file mode 100644
index 0000000..79ed386
Binary files /dev/null and b/project/sounds/aiy320000135.ogg differ
diff --git a/project/sounds/aiy320000140.ogg b/project/sounds/aiy320000140.ogg
new file mode 100644
index 0000000..0b7add2
Binary files /dev/null and b/project/sounds/aiy320000140.ogg differ
diff --git a/project/sounds/aiy340000010.ogg b/project/sounds/aiy340000010.ogg
new file mode 100644
index 0000000..1687505
Binary files /dev/null and b/project/sounds/aiy340000010.ogg differ
diff --git a/project/sounds/aiy340000020.ogg b/project/sounds/aiy340000020.ogg
new file mode 100644
index 0000000..f05a1e8
Binary files /dev/null and b/project/sounds/aiy340000020.ogg differ
diff --git a/project/sounds/aiy340000030.ogg b/project/sounds/aiy340000030.ogg
new file mode 100644
index 0000000..915b664
Binary files /dev/null and b/project/sounds/aiy340000030.ogg differ
diff --git a/project/sounds/aiy340000040.ogg b/project/sounds/aiy340000040.ogg
new file mode 100644
index 0000000..72259c5
Binary files /dev/null and b/project/sounds/aiy340000040.ogg differ
diff --git a/project/sounds/aiy340000050.ogg b/project/sounds/aiy340000050.ogg
new file mode 100644
index 0000000..ca136d2
Binary files /dev/null and b/project/sounds/aiy340000050.ogg differ
diff --git a/project/sounds/aiy340000060.ogg b/project/sounds/aiy340000060.ogg
new file mode 100644
index 0000000..8bcb94b
Binary files /dev/null and b/project/sounds/aiy340000060.ogg differ
diff --git a/project/sounds/aiy340000070.ogg b/project/sounds/aiy340000070.ogg
new file mode 100644
index 0000000..364be3b
Binary files /dev/null and b/project/sounds/aiy340000070.ogg differ
diff --git a/project/sounds/aiy340000080.ogg b/project/sounds/aiy340000080.ogg
new file mode 100644
index 0000000..6e6f1da
Binary files /dev/null and b/project/sounds/aiy340000080.ogg differ
diff --git a/project/sounds/aiy340000090.ogg b/project/sounds/aiy340000090.ogg
new file mode 100644
index 0000000..1da2a78
Binary files /dev/null and b/project/sounds/aiy340000090.ogg differ
diff --git a/project/sounds/aiy340000100.ogg b/project/sounds/aiy340000100.ogg
new file mode 100644
index 0000000..718db86
Binary files /dev/null and b/project/sounds/aiy340000100.ogg differ
diff --git a/project/sounds/aiy340000110.ogg b/project/sounds/aiy340000110.ogg
new file mode 100644
index 0000000..46d5be6
Binary files /dev/null and b/project/sounds/aiy340000110.ogg differ
diff --git a/project/sounds/aiy340000120.ogg b/project/sounds/aiy340000120.ogg
new file mode 100644
index 0000000..f1003c3
Binary files /dev/null and b/project/sounds/aiy340000120.ogg differ
diff --git a/project/sounds/aiy340000140.ogg b/project/sounds/aiy340000140.ogg
new file mode 100644
index 0000000..7a475e9
Binary files /dev/null and b/project/sounds/aiy340000140.ogg differ
diff --git a/project/sounds/aiy340000150.ogg b/project/sounds/aiy340000150.ogg
new file mode 100644
index 0000000..01b5107
Binary files /dev/null and b/project/sounds/aiy340000150.ogg differ
diff --git a/project/sounds/aiy340000160.ogg b/project/sounds/aiy340000160.ogg
new file mode 100644
index 0000000..710abca
Binary files /dev/null and b/project/sounds/aiy340000160.ogg differ
diff --git a/project/sounds/aiy340000170.ogg b/project/sounds/aiy340000170.ogg
new file mode 100644
index 0000000..7fc073b
Binary files /dev/null and b/project/sounds/aiy340000170.ogg differ
diff --git a/project/sounds/aiy340000180.ogg b/project/sounds/aiy340000180.ogg
new file mode 100644
index 0000000..575d8b4
Binary files /dev/null and b/project/sounds/aiy340000180.ogg differ
diff --git a/project/sounds/aiy340000190.ogg b/project/sounds/aiy340000190.ogg
new file mode 100644
index 0000000..155b411
Binary files /dev/null and b/project/sounds/aiy340000190.ogg differ
diff --git a/project/sounds/aiy340000200.ogg b/project/sounds/aiy340000200.ogg
new file mode 100644
index 0000000..bf6dc4a
Binary files /dev/null and b/project/sounds/aiy340000200.ogg differ
diff --git a/project/sounds/aiy340000210.ogg b/project/sounds/aiy340000210.ogg
new file mode 100644
index 0000000..3b4e1b1
Binary files /dev/null and b/project/sounds/aiy340000210.ogg differ
diff --git a/project/sounds/aiy340000220.ogg b/project/sounds/aiy340000220.ogg
new file mode 100644
index 0000000..1b46e59
Binary files /dev/null and b/project/sounds/aiy340000220.ogg differ
diff --git a/project/sounds/aiy340000230.ogg b/project/sounds/aiy340000230.ogg
new file mode 100644
index 0000000..a2462c7
Binary files /dev/null and b/project/sounds/aiy340000230.ogg differ
diff --git a/project/sounds/aiy340000240.ogg b/project/sounds/aiy340000240.ogg
new file mode 100644
index 0000000..91690e5
Binary files /dev/null and b/project/sounds/aiy340000240.ogg differ
diff --git a/project/sounds/aiy340000250.ogg b/project/sounds/aiy340000250.ogg
new file mode 100644
index 0000000..321f997
Binary files /dev/null and b/project/sounds/aiy340000250.ogg differ
diff --git a/project/sounds/aiy340000260.ogg b/project/sounds/aiy340000260.ogg
new file mode 100644
index 0000000..137c284
Binary files /dev/null and b/project/sounds/aiy340000260.ogg differ
diff --git a/project/sounds/aiy340000270.ogg b/project/sounds/aiy340000270.ogg
new file mode 100644
index 0000000..b94bbb4
Binary files /dev/null and b/project/sounds/aiy340000270.ogg differ
diff --git a/project/sounds/aiy340000280.ogg b/project/sounds/aiy340000280.ogg
new file mode 100644
index 0000000..e685598
Binary files /dev/null and b/project/sounds/aiy340000280.ogg differ
diff --git a/project/sounds/aiy340000290.ogg b/project/sounds/aiy340000290.ogg
new file mode 100644
index 0000000..6965511
Binary files /dev/null and b/project/sounds/aiy340000290.ogg differ
diff --git a/project/sounds/aiy340000300.ogg b/project/sounds/aiy340000300.ogg
new file mode 100644
index 0000000..f0c9ccd
Binary files /dev/null and b/project/sounds/aiy340000300.ogg differ
diff --git a/project/sounds/aiy340000310.ogg b/project/sounds/aiy340000310.ogg
new file mode 100644
index 0000000..886e7ba
Binary files /dev/null and b/project/sounds/aiy340000310.ogg differ
diff --git a/project/sounds/aiy340000320.ogg b/project/sounds/aiy340000320.ogg
new file mode 100644
index 0000000..c7b1ab1
Binary files /dev/null and b/project/sounds/aiy340000320.ogg differ
diff --git a/project/sounds/aiy340000330.ogg b/project/sounds/aiy340000330.ogg
new file mode 100644
index 0000000..b07b430
Binary files /dev/null and b/project/sounds/aiy340000330.ogg differ
diff --git a/project/sounds/aiy340000340.ogg b/project/sounds/aiy340000340.ogg
new file mode 100644
index 0000000..07fab33
Binary files /dev/null and b/project/sounds/aiy340000340.ogg differ
diff --git a/project/sounds/aiy340000350.ogg b/project/sounds/aiy340000350.ogg
new file mode 100644
index 0000000..4fff4de
Binary files /dev/null and b/project/sounds/aiy340000350.ogg differ
diff --git a/project/sounds/aiy340000360.ogg b/project/sounds/aiy340000360.ogg
new file mode 100644
index 0000000..1d9fa8f
Binary files /dev/null and b/project/sounds/aiy340000360.ogg differ
diff --git a/project/sounds/aiy340000370.ogg b/project/sounds/aiy340000370.ogg
new file mode 100644
index 0000000..a0e6e1e
Binary files /dev/null and b/project/sounds/aiy340000370.ogg differ
diff --git a/project/sounds/aiy340000380.ogg b/project/sounds/aiy340000380.ogg
new file mode 100644
index 0000000..0f4b917
Binary files /dev/null and b/project/sounds/aiy340000380.ogg differ
diff --git a/project/sounds/aiy340000390.ogg b/project/sounds/aiy340000390.ogg
new file mode 100644
index 0000000..021d193
Binary files /dev/null and b/project/sounds/aiy340000390.ogg differ
diff --git a/project/sounds/aiy340000400.ogg b/project/sounds/aiy340000400.ogg
new file mode 100644
index 0000000..3f217a7
Binary files /dev/null and b/project/sounds/aiy340000400.ogg differ
diff --git a/project/sounds/aiy340000410.ogg b/project/sounds/aiy340000410.ogg
new file mode 100644
index 0000000..95f46c9
Binary files /dev/null and b/project/sounds/aiy340000410.ogg differ
diff --git a/project/sounds/aiy340000420.ogg b/project/sounds/aiy340000420.ogg
new file mode 100644
index 0000000..7ed6067
Binary files /dev/null and b/project/sounds/aiy340000420.ogg differ
diff --git a/project/sounds/aiy340000430.ogg b/project/sounds/aiy340000430.ogg
new file mode 100644
index 0000000..cfc1202
Binary files /dev/null and b/project/sounds/aiy340000430.ogg differ
diff --git a/project/sounds/aiy350000240.ogg b/project/sounds/aiy350000240.ogg
new file mode 100644
index 0000000..e549bdd
Binary files /dev/null and b/project/sounds/aiy350000240.ogg differ
diff --git a/project/sounds/aiy350000250.ogg b/project/sounds/aiy350000250.ogg
new file mode 100644
index 0000000..3994c71
Binary files /dev/null and b/project/sounds/aiy350000250.ogg differ
diff --git a/project/sounds/aiy350000260.ogg b/project/sounds/aiy350000260.ogg
new file mode 100644
index 0000000..f8e3492
Binary files /dev/null and b/project/sounds/aiy350000260.ogg differ
diff --git a/project/sounds/aiy430000010.ogg b/project/sounds/aiy430000010.ogg
new file mode 100644
index 0000000..356ddcd
Binary files /dev/null and b/project/sounds/aiy430000010.ogg differ
diff --git a/project/sounds/aiy430000020.ogg b/project/sounds/aiy430000020.ogg
new file mode 100644
index 0000000..9efe052
Binary files /dev/null and b/project/sounds/aiy430000020.ogg differ
diff --git a/project/sounds/aiy430000030.ogg b/project/sounds/aiy430000030.ogg
new file mode 100644
index 0000000..c67ff9b
Binary files /dev/null and b/project/sounds/aiy430000030.ogg differ
diff --git a/project/sounds/aiy430000040.ogg b/project/sounds/aiy430000040.ogg
new file mode 100644
index 0000000..b05463e
Binary files /dev/null and b/project/sounds/aiy430000040.ogg differ
diff --git a/project/sounds/aiy430000050.ogg b/project/sounds/aiy430000050.ogg
new file mode 100644
index 0000000..5374b90
Binary files /dev/null and b/project/sounds/aiy430000050.ogg differ
diff --git a/project/sounds/aiy430000060.ogg b/project/sounds/aiy430000060.ogg
new file mode 100644
index 0000000..6b44c40
Binary files /dev/null and b/project/sounds/aiy430000060.ogg differ
diff --git a/project/sounds/aiy430000070.ogg b/project/sounds/aiy430000070.ogg
new file mode 100644
index 0000000..68fbcc8
Binary files /dev/null and b/project/sounds/aiy430000070.ogg differ
diff --git a/project/sounds/aiy430000080.ogg b/project/sounds/aiy430000080.ogg
new file mode 100644
index 0000000..f3ab8e1
Binary files /dev/null and b/project/sounds/aiy430000080.ogg differ
diff --git a/project/sounds/aiy430000090.ogg b/project/sounds/aiy430000090.ogg
new file mode 100644
index 0000000..a0487d9
Binary files /dev/null and b/project/sounds/aiy430000090.ogg differ
diff --git a/project/sounds/aiy430000100.ogg b/project/sounds/aiy430000100.ogg
new file mode 100644
index 0000000..3b111ed
Binary files /dev/null and b/project/sounds/aiy430000100.ogg differ
diff --git a/project/sounds/aiy430000110.ogg b/project/sounds/aiy430000110.ogg
new file mode 100644
index 0000000..472e8ea
Binary files /dev/null and b/project/sounds/aiy430000110.ogg differ
diff --git a/project/sounds/aiy430000120.ogg b/project/sounds/aiy430000120.ogg
new file mode 100644
index 0000000..16e11b0
Binary files /dev/null and b/project/sounds/aiy430000120.ogg differ
diff --git a/project/sounds/aiy430000130.ogg b/project/sounds/aiy430000130.ogg
new file mode 100644
index 0000000..87aaa6e
Binary files /dev/null and b/project/sounds/aiy430000130.ogg differ
diff --git a/project/sounds/aiy430000140.ogg b/project/sounds/aiy430000140.ogg
new file mode 100644
index 0000000..460a7de
Binary files /dev/null and b/project/sounds/aiy430000140.ogg differ
diff --git a/project/sounds/aiy430000150.ogg b/project/sounds/aiy430000150.ogg
new file mode 100644
index 0000000..a33a5bd
Binary files /dev/null and b/project/sounds/aiy430000150.ogg differ
diff --git a/project/sounds/aiy430000160.ogg b/project/sounds/aiy430000160.ogg
new file mode 100644
index 0000000..24971e2
Binary files /dev/null and b/project/sounds/aiy430000160.ogg differ
diff --git a/project/sounds/aiy430000170.ogg b/project/sounds/aiy430000170.ogg
new file mode 100644
index 0000000..a76c57b
Binary files /dev/null and b/project/sounds/aiy430000170.ogg differ
diff --git a/project/sounds/aiy430000180.ogg b/project/sounds/aiy430000180.ogg
new file mode 100644
index 0000000..aaf40ca
Binary files /dev/null and b/project/sounds/aiy430000180.ogg differ
diff --git a/project/sounds/aiy430000190.ogg b/project/sounds/aiy430000190.ogg
new file mode 100644
index 0000000..81ef542
Binary files /dev/null and b/project/sounds/aiy430000190.ogg differ
diff --git a/project/sounds/aiy430000200.ogg b/project/sounds/aiy430000200.ogg
new file mode 100644
index 0000000..57f8c8d
Binary files /dev/null and b/project/sounds/aiy430000200.ogg differ
diff --git a/project/sounds/aiy430000210.ogg b/project/sounds/aiy430000210.ogg
new file mode 100644
index 0000000..6a02c00
Binary files /dev/null and b/project/sounds/aiy430000210.ogg differ
diff --git a/project/sounds/aiy430000220.ogg b/project/sounds/aiy430000220.ogg
new file mode 100644
index 0000000..cc0e4be
Binary files /dev/null and b/project/sounds/aiy430000220.ogg differ
diff --git a/project/sounds/aiy430000230.ogg b/project/sounds/aiy430000230.ogg
new file mode 100644
index 0000000..a8ef1f3
Binary files /dev/null and b/project/sounds/aiy430000230.ogg differ
diff --git a/project/sounds/aiy430000240.ogg b/project/sounds/aiy430000240.ogg
new file mode 100644
index 0000000..6236846
Binary files /dev/null and b/project/sounds/aiy430000240.ogg differ
diff --git a/project/sounds/aiy430000250.ogg b/project/sounds/aiy430000250.ogg
new file mode 100644
index 0000000..806ecfb
Binary files /dev/null and b/project/sounds/aiy430000250.ogg differ
diff --git a/project/sounds/aiy430000260.ogg b/project/sounds/aiy430000260.ogg
new file mode 100644
index 0000000..8872913
Binary files /dev/null and b/project/sounds/aiy430000260.ogg differ
diff --git a/project/sounds/aiy430000270.ogg b/project/sounds/aiy430000270.ogg
new file mode 100644
index 0000000..2d61e3e
Binary files /dev/null and b/project/sounds/aiy430000270.ogg differ
diff --git a/project/sounds/aiy430000280.ogg b/project/sounds/aiy430000280.ogg
new file mode 100644
index 0000000..142bcbe
Binary files /dev/null and b/project/sounds/aiy430000280.ogg differ
diff --git a/project/sounds/aiy430000290.ogg b/project/sounds/aiy430000290.ogg
new file mode 100644
index 0000000..808500c
Binary files /dev/null and b/project/sounds/aiy430000290.ogg differ
diff --git a/project/sounds/aiy440000010.ogg b/project/sounds/aiy440000010.ogg
new file mode 100644
index 0000000..a8dad86
Binary files /dev/null and b/project/sounds/aiy440000010.ogg differ
diff --git a/project/sounds/aiy440000020.ogg b/project/sounds/aiy440000020.ogg
new file mode 100644
index 0000000..c9a14fd
Binary files /dev/null and b/project/sounds/aiy440000020.ogg differ
diff --git a/project/sounds/aiy440000030.ogg b/project/sounds/aiy440000030.ogg
new file mode 100644
index 0000000..fe311cc
Binary files /dev/null and b/project/sounds/aiy440000030.ogg differ
diff --git a/project/sounds/aiy440000040.ogg b/project/sounds/aiy440000040.ogg
new file mode 100644
index 0000000..2b98283
Binary files /dev/null and b/project/sounds/aiy440000040.ogg differ
diff --git a/project/sounds/aiy440000050.ogg b/project/sounds/aiy440000050.ogg
new file mode 100644
index 0000000..65e8f38
Binary files /dev/null and b/project/sounds/aiy440000050.ogg differ
diff --git a/project/sounds/aiy440000060.ogg b/project/sounds/aiy440000060.ogg
new file mode 100644
index 0000000..7911ca1
Binary files /dev/null and b/project/sounds/aiy440000060.ogg differ
diff --git a/project/sounds/aiy440000070.ogg b/project/sounds/aiy440000070.ogg
new file mode 100644
index 0000000..79ceecd
Binary files /dev/null and b/project/sounds/aiy440000070.ogg differ
diff --git a/project/sounds/aiy440000080.ogg b/project/sounds/aiy440000080.ogg
new file mode 100644
index 0000000..d165741
Binary files /dev/null and b/project/sounds/aiy440000080.ogg differ
diff --git a/project/sounds/aiy440000090.ogg b/project/sounds/aiy440000090.ogg
new file mode 100644
index 0000000..816a91f
Binary files /dev/null and b/project/sounds/aiy440000090.ogg differ
diff --git a/project/sounds/aiy440000100.ogg b/project/sounds/aiy440000100.ogg
new file mode 100644
index 0000000..e0cbe17
Binary files /dev/null and b/project/sounds/aiy440000100.ogg differ
diff --git a/project/sounds/aiy440000110.ogg b/project/sounds/aiy440000110.ogg
new file mode 100644
index 0000000..475cf95
Binary files /dev/null and b/project/sounds/aiy440000110.ogg differ
diff --git a/project/sounds/aiy440000120.ogg b/project/sounds/aiy440000120.ogg
new file mode 100644
index 0000000..ebdc162
Binary files /dev/null and b/project/sounds/aiy440000120.ogg differ
diff --git a/project/sounds/aiy510000010.ogg b/project/sounds/aiy510000010.ogg
new file mode 100644
index 0000000..5ab3fc0
Binary files /dev/null and b/project/sounds/aiy510000010.ogg differ
diff --git a/project/sounds/aiy510000020.ogg b/project/sounds/aiy510000020.ogg
new file mode 100644
index 0000000..57f77d7
Binary files /dev/null and b/project/sounds/aiy510000020.ogg differ
diff --git a/project/sounds/aiy720000010.ogg b/project/sounds/aiy720000010.ogg
new file mode 100644
index 0000000..5ae9b67
Binary files /dev/null and b/project/sounds/aiy720000010.ogg differ
diff --git a/project/sounds/aiy730000010.ogg b/project/sounds/aiy730000010.ogg
new file mode 100644
index 0000000..ab0d410
Binary files /dev/null and b/project/sounds/aiy730000010.ogg differ
diff --git a/project/sounds/aiy740000010.ogg b/project/sounds/aiy740000010.ogg
new file mode 100644
index 0000000..b5b010f
Binary files /dev/null and b/project/sounds/aiy740000010.ogg differ
diff --git a/project/sounds/aiy750000010.ogg b/project/sounds/aiy750000010.ogg
new file mode 100644
index 0000000..0b8ceb4
Binary files /dev/null and b/project/sounds/aiy750000010.ogg differ
diff --git a/project/sounds/aiy750000020.ogg b/project/sounds/aiy750000020.ogg
new file mode 100644
index 0000000..f260c12
Binary files /dev/null and b/project/sounds/aiy750000020.ogg differ
diff --git a/project/sounds/aiy750000030.ogg b/project/sounds/aiy750000030.ogg
new file mode 100644
index 0000000..93faa55
Binary files /dev/null and b/project/sounds/aiy750000030.ogg differ
diff --git a/project/sounds/aiy750000040.ogg b/project/sounds/aiy750000040.ogg
new file mode 100644
index 0000000..7592d4b
Binary files /dev/null and b/project/sounds/aiy750000040.ogg differ
diff --git a/project/sounds/aiy750000050.ogg b/project/sounds/aiy750000050.ogg
new file mode 100644
index 0000000..bf95779
Binary files /dev/null and b/project/sounds/aiy750000050.ogg differ
diff --git a/project/sounds/aiy750000060.ogg b/project/sounds/aiy750000060.ogg
new file mode 100644
index 0000000..b0f81b8
Binary files /dev/null and b/project/sounds/aiy750000060.ogg differ
diff --git a/project/sounds/aiy750000070.ogg b/project/sounds/aiy750000070.ogg
new file mode 100644
index 0000000..d2edb4d
Binary files /dev/null and b/project/sounds/aiy750000070.ogg differ
diff --git a/project/sounds/aiy750000080.ogg b/project/sounds/aiy750000080.ogg
new file mode 100644
index 0000000..a0ffff8
Binary files /dev/null and b/project/sounds/aiy750000080.ogg differ
diff --git a/project/sounds/aiy750000090.ogg b/project/sounds/aiy750000090.ogg
new file mode 100644
index 0000000..aefcdb3
Binary files /dev/null and b/project/sounds/aiy750000090.ogg differ
diff --git a/project/sounds/aiy750000100.ogg b/project/sounds/aiy750000100.ogg
new file mode 100644
index 0000000..4057d0c
Binary files /dev/null and b/project/sounds/aiy750000100.ogg differ
diff --git a/project/sounds/aiy750000110.ogg b/project/sounds/aiy750000110.ogg
new file mode 100644
index 0000000..e76d3be
Binary files /dev/null and b/project/sounds/aiy750000110.ogg differ
diff --git a/project/sounds/aiy750000120.ogg b/project/sounds/aiy750000120.ogg
new file mode 100644
index 0000000..41d7788
Binary files /dev/null and b/project/sounds/aiy750000120.ogg differ
diff --git a/project/sounds/aiy750000130.ogg b/project/sounds/aiy750000130.ogg
new file mode 100644
index 0000000..161f900
Binary files /dev/null and b/project/sounds/aiy750000130.ogg differ
diff --git a/project/sounds/aiy750000140.ogg b/project/sounds/aiy750000140.ogg
new file mode 100644
index 0000000..fda459c
Binary files /dev/null and b/project/sounds/aiy750000140.ogg differ
diff --git a/project/sounds/aiy750000150.ogg b/project/sounds/aiy750000150.ogg
new file mode 100644
index 0000000..8422fce
Binary files /dev/null and b/project/sounds/aiy750000150.ogg differ
diff --git a/project/sounds/aiy750000160.ogg b/project/sounds/aiy750000160.ogg
new file mode 100644
index 0000000..62f8163
Binary files /dev/null and b/project/sounds/aiy750000160.ogg differ
diff --git a/project/sounds/aiy750000170.ogg b/project/sounds/aiy750000170.ogg
new file mode 100644
index 0000000..b9fe06e
Binary files /dev/null and b/project/sounds/aiy750000170.ogg differ
diff --git a/project/sounds/aiy750000180.ogg b/project/sounds/aiy750000180.ogg
new file mode 100644
index 0000000..b7fb85f
Binary files /dev/null and b/project/sounds/aiy750000180.ogg differ
diff --git a/project/sounds/aiy750000190.ogg b/project/sounds/aiy750000190.ogg
new file mode 100644
index 0000000..1d7b4a8
Binary files /dev/null and b/project/sounds/aiy750000190.ogg differ
diff --git a/project/sounds/aiy750000200.ogg b/project/sounds/aiy750000200.ogg
new file mode 100644
index 0000000..68d41da
Binary files /dev/null and b/project/sounds/aiy750000200.ogg differ
diff --git a/project/sounds/aiy750000210.ogg b/project/sounds/aiy750000210.ogg
new file mode 100644
index 0000000..faf634b
Binary files /dev/null and b/project/sounds/aiy750000210.ogg differ
diff --git a/project/sounds/aiy750000220.ogg b/project/sounds/aiy750000220.ogg
new file mode 100644
index 0000000..1b931cf
Binary files /dev/null and b/project/sounds/aiy750000220.ogg differ
diff --git a/project/sounds/aiy750000230.ogg b/project/sounds/aiy750000230.ogg
new file mode 100644
index 0000000..3c31602
Binary files /dev/null and b/project/sounds/aiy750000230.ogg differ
diff --git a/project/sounds/aiy750000240.ogg b/project/sounds/aiy750000240.ogg
new file mode 100644
index 0000000..ad8501c
Binary files /dev/null and b/project/sounds/aiy750000240.ogg differ
diff --git a/project/sounds/aiy750000250.ogg b/project/sounds/aiy750000250.ogg
new file mode 100644
index 0000000..3ebaffe
Binary files /dev/null and b/project/sounds/aiy750000250.ogg differ
diff --git a/project/sounds/aiy750000260.ogg b/project/sounds/aiy750000260.ogg
new file mode 100644
index 0000000..c45ca4d
Binary files /dev/null and b/project/sounds/aiy750000260.ogg differ
diff --git a/project/sounds/aiy750000270.ogg b/project/sounds/aiy750000270.ogg
new file mode 100644
index 0000000..ad0c25d
Binary files /dev/null and b/project/sounds/aiy750000270.ogg differ
diff --git a/project/sounds/aiy750000280.ogg b/project/sounds/aiy750000280.ogg
new file mode 100644
index 0000000..2b1a3bc
Binary files /dev/null and b/project/sounds/aiy750000280.ogg differ
diff --git a/project/sounds/aiy750000290.ogg b/project/sounds/aiy750000290.ogg
new file mode 100644
index 0000000..11d69ae
Binary files /dev/null and b/project/sounds/aiy750000290.ogg differ
diff --git a/project/sounds/aiy750000300.ogg b/project/sounds/aiy750000300.ogg
new file mode 100644
index 0000000..145e3b3
Binary files /dev/null and b/project/sounds/aiy750000300.ogg differ
diff --git a/project/sounds/aiy750000310.ogg b/project/sounds/aiy750000310.ogg
new file mode 100644
index 0000000..aac984e
Binary files /dev/null and b/project/sounds/aiy750000310.ogg differ
diff --git a/project/sounds/aiy750000320.ogg b/project/sounds/aiy750000320.ogg
new file mode 100644
index 0000000..175ab60
Binary files /dev/null and b/project/sounds/aiy750000320.ogg differ
diff --git a/project/sounds/aiy750000330.ogg b/project/sounds/aiy750000330.ogg
new file mode 100644
index 0000000..dd13862
Binary files /dev/null and b/project/sounds/aiy750000330.ogg differ
diff --git a/project/sounds/aiy750000340.ogg b/project/sounds/aiy750000340.ogg
new file mode 100644
index 0000000..46c6c87
Binary files /dev/null and b/project/sounds/aiy750000340.ogg differ
diff --git a/project/sounds/aiy800000010.ogg b/project/sounds/aiy800000010.ogg
new file mode 100644
index 0000000..f7c1058
Binary files /dev/null and b/project/sounds/aiy800000010.ogg differ
diff --git a/project/sounds/aiy800000020.ogg b/project/sounds/aiy800000020.ogg
new file mode 100644
index 0000000..e5233c7
Binary files /dev/null and b/project/sounds/aiy800000020.ogg differ
diff --git a/project/sounds/aiy800000030.ogg b/project/sounds/aiy800000030.ogg
new file mode 100644
index 0000000..4b9d26c
Binary files /dev/null and b/project/sounds/aiy800000030.ogg differ
diff --git a/project/sounds/aiy800000040.ogg b/project/sounds/aiy800000040.ogg
new file mode 100644
index 0000000..64cca36
Binary files /dev/null and b/project/sounds/aiy800000040.ogg differ
diff --git a/project/sounds/aiy800000050.ogg b/project/sounds/aiy800000050.ogg
new file mode 100644
index 0000000..20c4b22
Binary files /dev/null and b/project/sounds/aiy800000050.ogg differ
diff --git a/project/sounds/aiy800000060.ogg b/project/sounds/aiy800000060.ogg
new file mode 100644
index 0000000..baf6f0a
Binary files /dev/null and b/project/sounds/aiy800000060.ogg differ
diff --git a/project/sounds/aiy800000070.ogg b/project/sounds/aiy800000070.ogg
new file mode 100644
index 0000000..09b6793
Binary files /dev/null and b/project/sounds/aiy800000070.ogg differ
diff --git a/project/sounds/aiy800000080.ogg b/project/sounds/aiy800000080.ogg
new file mode 100644
index 0000000..42f1925
Binary files /dev/null and b/project/sounds/aiy800000080.ogg differ
diff --git a/project/sounds/aiy800000090.ogg b/project/sounds/aiy800000090.ogg
new file mode 100644
index 0000000..0d1df0d
Binary files /dev/null and b/project/sounds/aiy800000090.ogg differ
diff --git a/project/sounds/aiy800000100.ogg b/project/sounds/aiy800000100.ogg
new file mode 100644
index 0000000..6f1ecbc
Binary files /dev/null and b/project/sounds/aiy800000100.ogg differ
diff --git a/project/sounds/aiy810000010.ogg b/project/sounds/aiy810000010.ogg
new file mode 100644
index 0000000..141c976
Binary files /dev/null and b/project/sounds/aiy810000010.ogg differ
diff --git a/project/sounds/aiy810000020.ogg b/project/sounds/aiy810000020.ogg
new file mode 100644
index 0000000..f755114
Binary files /dev/null and b/project/sounds/aiy810000020.ogg differ
diff --git a/project/sounds/aiy810000030.ogg b/project/sounds/aiy810000030.ogg
new file mode 100644
index 0000000..ece7da3
Binary files /dev/null and b/project/sounds/aiy810000030.ogg differ
diff --git a/project/sounds/aiy810000040.ogg b/project/sounds/aiy810000040.ogg
new file mode 100644
index 0000000..d600ea7
Binary files /dev/null and b/project/sounds/aiy810000040.ogg differ
diff --git a/project/sounds/aiy810000050.ogg b/project/sounds/aiy810000050.ogg
new file mode 100644
index 0000000..b3975f8
Binary files /dev/null and b/project/sounds/aiy810000050.ogg differ
diff --git a/project/sounds/aiy810000060.ogg b/project/sounds/aiy810000060.ogg
new file mode 100644
index 0000000..f6cf315
Binary files /dev/null and b/project/sounds/aiy810000060.ogg differ
diff --git a/project/sounds/aiy810000070.ogg b/project/sounds/aiy810000070.ogg
new file mode 100644
index 0000000..0944056
Binary files /dev/null and b/project/sounds/aiy810000070.ogg differ
diff --git a/project/sounds/aiy860000010.ogg b/project/sounds/aiy860000010.ogg
new file mode 100644
index 0000000..d6ffe53
Binary files /dev/null and b/project/sounds/aiy860000010.ogg differ
diff --git a/project/sounds/aiy860000020.ogg b/project/sounds/aiy860000020.ogg
new file mode 100644
index 0000000..8157a6f
Binary files /dev/null and b/project/sounds/aiy860000020.ogg differ
diff --git a/project/sounds/aiy860000030.ogg b/project/sounds/aiy860000030.ogg
new file mode 100644
index 0000000..f428615
Binary files /dev/null and b/project/sounds/aiy860000030.ogg differ
diff --git a/project/sounds/aiy860000040.ogg b/project/sounds/aiy860000040.ogg
new file mode 100644
index 0000000..cd1a1eb
Binary files /dev/null and b/project/sounds/aiy860000040.ogg differ
diff --git a/project/sounds/aiy860000050.ogg b/project/sounds/aiy860000050.ogg
new file mode 100644
index 0000000..e0cf702
Binary files /dev/null and b/project/sounds/aiy860000050.ogg differ
diff --git a/project/sounds/aiy860000060.ogg b/project/sounds/aiy860000060.ogg
new file mode 100644
index 0000000..c540697
Binary files /dev/null and b/project/sounds/aiy860000060.ogg differ
diff --git a/project/tilesets/2233.png b/project/tilesets/2233.png
new file mode 100644
index 0000000..a3bba4f
Binary files /dev/null and b/project/tilesets/2233.png differ
diff --git a/project/tilesets/C5.webp b/project/tilesets/C5.webp
index 641de3c..bbb361e 100644
Binary files a/project/tilesets/C5.webp and b/project/tilesets/C5.webp differ
diff --git a/project/tilesets/c6.png b/project/tilesets/c6.png
new file mode 100644
index 0000000..ecec049
Binary files /dev/null and b/project/tilesets/c6.png differ
diff --git a/styles.css b/styles.css
index 89ddc05..139471b 100644
--- a/styles.css
+++ b/styles.css
@@ -1,534 +1,618 @@
-html, body {
-    margin: 0;
-    padding: 0;
-    width: 100%;
-    height: 100%;
-    background-color: #000;
-    overflow: hidden;
+html,
+body {
+  margin: 0;
+  padding: 0;
+  width: 100%;
+  height: 100%;
+  background-color: #000;
+  overflow: hidden;
 }
 
 #gameGroup {
-    position: absolute;
-    box-sizing: border-box;
-    -moz-box-sizing: border-box;
-    -webkit-box-sizing: border-box;
-    background-color: #000;
+  position: absolute;
+  box-sizing: border-box;
+  -moz-box-sizing: border-box;
+  -webkit-box-sizing: border-box;
+  background-color: #000;
 }
 
 #mainTips {
-    color: #fff;
-    font-size: 0.8em;
-    position: fixed;
-    top: 10px;
-    left: 10px;
-    z-index: 370;
+  color: #fff;
+  font-size: 0.8em;
+  position: fixed;
+  top: 10px;
+  left: 10px;
+  z-index: 370;
 }
 
 #musicBtn {
-    position: fixed;
-    bottom: 3px;
-    right: 3px;
-    cursor: pointer;
-    z-index: 400;
-    display: none;
+  position: fixed;
+  bottom: 3px;
+  right: 3px;
+  cursor: pointer;
+  z-index: 400;
+  display: none;
 }
 
 #enlargeBtn {
-    position: fixed;
-    bottom: 3px;
-    right: 34px;
-    cursor: pointer;
-    z-index: 400;
-    display: none;
+  position: fixed;
+  bottom: 3px;
+  right: 34px;
+  cursor: pointer;
+  z-index: 400;
+  display: none;
 }
 
 #startPanel {
-    width: 100%;
-    height: 100%;
-    position: absolute;
-    top: 0;
-    left: 0;
-    background-color: #fff;
-    overflow: hidden;
-    z-index: 300;
+  width: 100%;
+  height: 100%;
+  position: absolute;
+  top: 0;
+  left: 0;
+  background-color: #fff;
+  overflow: hidden;
+  z-index: 300;
 }
 
 #startTop {
-    width: 100%;
-    height: 100%;
-    position: absolute;
-    top: 0;
-    left: 0;
-    background-color: #000;
-    z-index: 350;
+  width: 100%;
+  height: 100%;
+  position: absolute;
+  top: 0;
+  left: 0;
+  background-color: #000;
+  z-index: 350;
 }
 
 #startTopProgressBar {
-    width: 90%;
-    height: 5%;
-    margin: 0 5%;
-    position: absolute;
-    top: 5%;
-    background-color: #fff;
-    z-index: 15;
+  width: 90%;
+  height: 5%;
+  margin: 0 5%;
+  position: absolute;
+  top: 5%;
+  background-color: #fff;
+  z-index: 15;
 }
 
 #startTopProgress {
-    width: 0%;
-    height: 100%;
-    background-color: #666;
+  width: 0%;
+  height: 100%;
+  background-color: #666;
 }
 
 #startTopLoadTips {
-    color: #fff;
-    font-size: 0.6em;
-    position: absolute;
-    top: 10%;
-    left: 5%;
-    z-index: 15;
+  color: #fff;
+  font-size: 0.6em;
+  position: absolute;
+  top: 10%;
+  left: 5%;
+  z-index: 15;
 }
 
 #startTopHint {
-    color: #66CCFF;
-    position: absolute;
-    bottom: 0;
-    left: 5%;
-    z-index: 15;
-    font-size: 1.1em;
+  color: #66ccff;
+  position: absolute;
+  bottom: 0;
+  left: 5%;
+  z-index: 15;
+  font-size: 1.1em;
 }
 
 #startBackground {
-    position:absolute;
-    top:50%;
-    left:50%;
-    height: 100%;
-    width: 100%;
-    transform:translate(-50%,-50%);
-    z-index: 260;
+  position: absolute;
+  top: 50%;
+  left: 50%;
+  height: 100%;
+  width: 100%;
+  transform: translate(-50%, -50%);
+  z-index: 260;
 }
 
 #startLogo {
-    user-select: none;
-    position: absolute;
-    z-index: 290;
-    left: 0;
-    right: 0;
-    margin-left: auto;
-    margin-right: auto;
-    margin-top: 8%;
-    max-width: 100%;
-    text-align: center;
-    font: bold 0em STXingkai;
-	color: #000000;
+  user-select: none;
+  position: absolute;
+  z-index: 290;
+  left: 0;
+  right: 0;
+  margin-left: auto;
+  margin-right: auto;
+  margin-top: 8%;
+  max-width: 100%;
+  text-align: center;
+  font: bold 0em STXingkai;
+  color: #000000;
 }
 
 #startTitle {
-    user-select: none;
-    position: absolute;
-    z-index: 280;
+  user-select: none;
+  position: absolute;
+  z-index: 280;
 }
 
 #startButtonGroup {
-    width: auto;
-    position: absolute;
-    text-align: center;
-    font-size: 1.4em;
-    display: none;
-    z-index: 310;
-    bottom: 0;
-    margin-bottom: 5%;
-    left: 50%;
-    transform: translateX(-50%);
-    padding: 15px 25px;
-    min-width: 100%;
-    /* default value */
-    background-color: #000000;
-    opacity: 0.55;
-    color: #FFFFFF;
-    border: #FFFFFF 0px solid;
-    caret-color: #ffffff;
-    border-radius: 0px;
+  width: auto;
+  position: absolute;
+  text-align: center;
+  font-size: 1.4em;
+  display: none;
+  z-index: 310;
+  bottom: 0;
+  margin-bottom: 5%;
+  left: 50%;
+  transform: translateX(-50%);
+  padding: 15px 25px;
+  min-width: 100%;
+  /* default value */
+  background-color: #000000;
+  opacity: 0.55;
+  color: #ffffff;
+  border: #ffffff 0px solid;
+  caret-color: #ffffff;
+  border-radius: 0px;
 }
 
 #startButtons {
-    display: none;
+  display: none;
 }
 
 #levelChooseButtons {
-    display: none;
+  display: none;
 }
 
 .startButton {
-    width: 100%;
-    margin: 0;
-	font:1.2rem "pala", sans-serif;
-    font-weight: normal;
-    display: block;
-    cursor: pointer;
-    padding: 0px 0;
-    border-color: transparent;
-    border-width: 0px;
-    border-style: solid;
-    border-radius: 0px;
+  width: 100%;
+  margin: 0;
+  font: 1.2rem pala, sans-serif;
+  font-weight: normal;
+  display: block;
+  cursor: pointer;
+  padding: 0px 0;
+  border-color: transparent;
+  border-width: 0px;
+  border-style: solid;
+  border-radius: 0px;
 }
 
 .onChoiceAnimate {
-    animation: onChoice 2s ease-in-out 0s infinite normal none running;
+  animation: onChoice 2s ease-in-out 0s infinite normal none running;
 }
 
 #floorMsgGroup {
-    position: absolute;
-    text-align: center;
-    display: none;
-    color: #fff;
-    background-color: #000;
-    z-index: 230;
+  position: absolute;
+  text-align: center;
+  display: none;
+  color: #fff;
+  background-color: #000;
+  z-index: 230;
 }
 
 #logoLabel {
-    margin-top: 8%;
-    font: bold 3em STXingkai;
-    margin-left: auto;
-    margin-right: auto;
+  margin-top: 8%;
+  font: bold 3em STXingkai;
+  margin-left: auto;
+  margin-right: auto;
 }
 
 #versionLabel {
-    margin-top: -3%;
-    font-size: 1.2em;
-    font-weight: bold;
+  margin-top: -3%;
+  font-size: 1.2em;
+  font-weight: bold;
 }
 
 #floorNameLabel {
-    margin-top: 30px;
-    font-size: 1.6em;
-    font-weight: bold;
+  margin-top: 30px;
+  font-size: 1.6em;
+  font-weight: bold;
 }
 
 #statusBar {
-    position: absolute;
-    box-sizing: border-box;
-    -moz-box-sizing: border-box;
-    -webkit-box-sizing: border-box;
-    background: url(project/materials/ground.png) repeat;
-    z-index: 185;
-    display: none;
-    top: 0;
-    left: 0;
-    padding: 3px;
+  position: absolute;
+  box-sizing: border-box;
+  -moz-box-sizing: border-box;
+  -webkit-box-sizing: border-box;
+  background: url(project/materials/ground.png) repeat;
+  z-index: 185;
+  display: none;
+  top: 0;
+  left: 0;
+  padding: 3px;
 }
-#statusBar .status{
-    position: relative;
-    display: block;
-    float: left;
-    width: 100%;
+#statusBar .status {
+  position: relative;
+  display: block;
+  float: left;
+  width: 100%;
 }
-.status img{
-    vertical-align: middle;
-    width: auto;
-    height: 100%;
-    max-height: 1.6em;
+.status img {
+  vertical-align: middle;
+  width: auto;
+  height: 100%;
+  max-height: 1.6em;
 }
-#statusBar span{
-    font: bold italic 1.1em Verdana;
-    display: inline;
+#statusBar span {
+  font: bold italic 1.1em Verdana;
+  display: inline;
 }
 #statusBar p {
-    display: inline-block;
-    vertical-align: middle;
-    width: 60%;
-    margin: 0;
-    color: white;
-    font: bold italic 1.1em Verdana;
-    white-space: nowrap;
+  display: inline-block;
+  vertical-align: middle;
+  width: 60%;
+  margin: 0;
+  color: white;
+  font: bold italic 1.1em Verdana;
+  white-space: nowrap;
 }
 #toolBar {
-    position: absolute;
-    background: url(project/materials/ground.png) repeat;
-    z-index: 210;
-    box-sizing: border-box;
-    -moz-box-sizing: border-box;
-    -webkit-box-sizing: border-box;
-    display: none;
-    padding: 3px;
+  position: absolute;
+  background: url(project/materials/ground.png) repeat;
+  z-index: 210;
+  box-sizing: border-box;
+  -moz-box-sizing: border-box;
+  -webkit-box-sizing: border-box;
+  display: none;
+  padding: 3px;
 }
-#toolBar .tools{
-    position: relative;
-    display: block;
-    float: left;
+#toolBar .tools {
+  position: relative;
+  display: block;
+  float: left;
 }
 
 p#hard {
-    width: 6em;
-    vertical-align: middle;
-    display: inline-block;
-    color: red;
-    font: bold normal 1.1em "Arial Black";
-    text-align: center;
-    margin: 0 6px 6px 0;
-    word-break: keep-all;
+  width: 6em;
+  vertical-align: middle;
+  display: inline-block;
+  color: red;
+  font: bold normal 1.1em "Arial Black";
+  text-align: center;
+  margin: 0 6px 6px 0;
+  word-break: keep-all;
 }
 
-span#poison, span#weak, span#curse, span#pickaxe, span#bomb, span#fly {
-    font-style: normal;
-    font-size: 1em;
+span#poison,
+span#weak,
+span#curse,
+span#pickaxe,
+span#bomb,
+span#fly {
+  font-style: normal;
+  font-size: 1em;
 }
 
 p#name {
-    font-style: normal;
+  font-style: normal;
 }
 
 .gameCanvas {
-    position: absolute;
-    box-sizing: border-box;
-    -moz-box-sizing: border-box;
-    -webkit-box-sizing: border-box;
+  position: absolute;
+  box-sizing: border-box;
+  -moz-box-sizing: border-box;
+  -webkit-box-sizing: border-box;
 }
 
 #gif {
-    z-index: 20;
-    position: absolute;
-    overflow: hidden;
+  z-index: 20;
+  position: absolute;
+  overflow: hidden;
 }
 
 #gif2 {
-    z-index: 90;
-    position: absolute;
-    overflow: hidden;
+  z-index: 90;
+  position: absolute;
+  overflow: hidden;
 }
 
 #gameDraw {
-    position: absolute;
-    background: #000000;
-    overflow: hidden;
-    z-index: 185;
+  position: absolute;
+  background: #000000;
+  overflow: hidden;
+  z-index: 185;
 }
 
 #bg {
-    z-index: 10;
+  z-index: 10;
 }
 
 #event {
-    z-index: 30;
+  z-index: 30;
 }
 
 #hero {
-    z-index: 40;
+  z-index: 40;
 }
 
 #event2 {
-    z-index: 50;
+  z-index: 50;
 }
 
 #fg {
-    z-index: 60;
+  z-index: 60;
 }
 
 #damage {
-    z-index: 65;
+  z-index: 65;
 }
 
 #animate {
-    z-index: 70;
+  z-index: 70;
 }
 
 #curtain {
-    z-index: 125;
+  z-index: 125;
 }
 
 #ui {
-    z-index: 140;
+  z-index: 140;
 }
 
 #data {
-    z-index: 170;
+  z-index: 170;
 }
 
 #inputDiv {
-    display: none;
-    width: 100%;
-    height: 100%;
-    position: fixed;
-    top: 0;
-    left: 0;
-    background: rgba(127,127,127,0.6);
-    z-index: 2000
+  display: none;
+  width: 100%;
+  height: 100%;
+  position: fixed;
+  top: 0;
+  left: 0;
+  background: rgba(127, 127, 127, 0.6);
+  z-index: 2000;
 }
 
 #inputDialog {
-    position: fixed;
-    top: 50%;
-    left: 50%;
-    transform: translate(-50%, -55%);
-    background: white;
-    width: 250px;
-    min-height: 50px;
+  position: fixed;
+  top: 50%;
+  left: 50%;
+  transform: translate(-50%, -55%);
+  background: white;
+  width: 250px;
+  min-height: 50px;
 }
 
 #inputMessage {
-    word-wrap: break-word;
-    text-align: left;
-    margin-left: 8%;
-    margin-right: 5%;
+  word-wrap: break-word;
+  text-align: left;
+  margin-left: 8%;
+  margin-right: 5%;
 }
 
 #inputBox {
-    margin-left: 8%;
-    width: 80%;
-    margin-bottom: 10px;
-    padding: 5px 3px;
-    border: 1px solid;
-    background: #F0F0F0;
+  margin-left: 8%;
+  width: 80%;
+  margin-bottom: 10px;
+  padding: 5px 3px;
+  border: 1px solid;
+  background: #f0f0f0;
 }
 
 #inputYes {
-    margin-bottom: 15px;
-    margin-left: 8%;
+  margin-bottom: 15px;
+  margin-left: 8%;
 }
 
 #inputNo {
-    float:right;
-    margin-right: 10%;
+  float: right;
+  margin-right: 10%;
 }
 
-#_selector, ._uievent_selector {
-    animation: selector 2s ease-in-out 0s infinite normal none running;
+#_selector,
+._uievent_selector {
+  animation: selector 2s ease-in-out 0s infinite normal none running;
 }
 
 @-webkit-keyframes selector {
-    0% { opacity: 0.95; }
-    50% { opacity: 0.55; }
-    100% { opacity: 0.95; }
+  0% {
+    opacity: 0.95;
+  }
+  50% {
+    opacity: 0.55;
+  }
+  100% {
+    opacity: 0.95;
+  }
 }
 
 @keyframes selector {
-    0% { opacity: 0.95; }
-    50% { opacity: 0.55; }
-    100% { opacity: 0.95; }
+  0% {
+    opacity: 0.95;
+  }
+  50% {
+    opacity: 0.55;
+  }
+  100% {
+    opacity: 0.95;
+  }
 }
 
 #next {
-    width: 5px;
-    height: 5px;
-    display: none;
-    position: absolute;
-    transform: rotate(45deg);
-    border-bottom-width: 4px;
-    border-bottom-style: solid;
-    border-right-width: 4px;
-    border-right-style: solid;
-    -webkit-animation: next .5s ease-in-out alternate infinite;
-    animation: next .5s ease-in-out alternate infinite;
-    left: 0;
-    top: 0;
-    opacity: 0.7;
-    z-index: 169;
+  width: 5px;
+  height: 5px;
+  display: none;
+  position: absolute;
+  transform: rotate(45deg);
+  border-bottom-width: 4px;
+  border-bottom-style: solid;
+  border-right-width: 4px;
+  border-right-style: solid;
+  -webkit-animation: next 0.5s ease-in-out alternate infinite;
+  animation: next 0.5s ease-in-out alternate infinite;
+  left: 0;
+  top: 0;
+  opacity: 0.7;
+  z-index: 169;
 }
 
 @-webkit-keyframes next {
-    100% {
-        transform: rotate(45deg) translate(-3px, -3px);
-    }
+  100% {
+    transform: rotate(45deg) translate(-3px, -3px);
+  }
 }
 @keyframes next {
-    100% {
-        transform: rotate(45deg) translate(-3px, -3px);
-    }
+  100% {
+    transform: rotate(45deg) translate(-3px, -3px);
+  }
 }
 
 #startImageBackgroundDiv {
-    display: none;
-    width: 100%;
-    height: 100%;
-    position: fixed;
-    z-index: 10000;
+  display: none;
+  width: 100%;
+  height: 100%;
+  position: fixed;
+  z-index: 10000;
 }
 
 #startImageDiv {
-    width: 100%;
-    height: 100%;
-    position: fixed;
-    background: black;
-    opacity: 1;
+  width: 100%;
+  height: 100%;
+  position: fixed;
+  background: black;
+  opacity: 1;
 }
 
 #startImageLogo {
-    opacity: 0;
-    max-width: 60%;
-    max-height: 60%;
-    position: fixed;
-    left: 50%;
-    top: 50%;
-    transform: translate(-50%, -50%);
+  opacity: 0;
+  max-width: 60%;
+  max-height: 60%;
+  position: fixed;
+  left: 50%;
+  top: 50%;
+  transform: translate(-50%, -50%);
 }
 
 .startImageAnimation {
-    -webkit-animation: startImage 4s ease-in-out 1s alternate 1;
-    animation: startImage 4s ease-in-out 1s alternate 1;
+  -webkit-animation: startImage 4s ease-in-out 1s alternate 1;
+  animation: startImage 4s ease-in-out 1s alternate 1;
 }
 
 @-webkit-keyframes startImage {
-    0% { opacity: 0; }
-    60% { opacity: 1; }
-    100% { opacity: 0; }
+  0% {
+    opacity: 0;
+  }
+  60% {
+    opacity: 1;
+  }
+  100% {
+    opacity: 0;
+  }
 }
 
 @keyframes startImage {
-    0% { opacity: 0; }
-    60% { opacity: 1; }
-    100% { opacity: 0; }
+  0% {
+    opacity: 0;
+  }
+  60% {
+    opacity: 1;
+  }
+  100% {
+    opacity: 0;
+  }
 }
 
 .startImageDivAnimation {
-    -webkit-animation: startImageDivDisappear 2s ease-in-out 5s alternate 1;
-    animation: startImageDivDisappear 2s ease-in-out 5s alternate 1;
+  -webkit-animation: startImageDivDisappear 2s ease-in-out 5s alternate 1;
+  animation: startImageDivDisappear 2s ease-in-out 5s alternate 1;
 }
 
 @-webkit-keyframes startImageDivDisappear {
-    0% { opacity: 1 }
-    100% { opacity: 0 }
+  0% {
+    opacity: 1;
+  }
+  100% {
+    opacity: 0;
+  }
 }
 
 @keyframes startImageDivDisappear {
-    0% { opacity: 1 }
-    100% { opacity: 0 }
+  0% {
+    opacity: 1;
+  }
+  100% {
+    opacity: 0;
+  }
 }
 #ui-editor {
-    z-index: 9999;
-    position: absolute;
-    overflow: visible;
-    height: 100%;
+  z-index: 9999;
+  position: absolute;
+  overflow: visible;
+  height: 100%;
 }
 
 @font-face {
-    font-family: Fira Code;
-    src: url(../src/fonts/FiraCode-Regular.ttf);
+  font-family: Fira Code;
+  src: url(../src/fonts/FiraCode-Regular.ttf);
 }
 
 /* 注释下面这三行以开启抗锯齿 */
-.anti-aliasing {
-    image-rendering: pixelated;
-}
+/* .anti-aliasing {
+  image-rendering: pixelated;
+} */
 
 .warning {
-    transition: left cubic-bezier(0, 0.9, 1, 0.1) 2.5s;
-    -webkit-transition: left cubic-bezier(0, 0.9, 1, 0.1) 2.5s;
-    animation: warning 1s linear 0s alternate infinite;
-    -webkit-animation: warning 1s linear 0s alternate infinite;
+  transition: left cubic-bezier(0, 0.9, 1, 0.1) 2.5s;
+  -webkit-transition: left cubic-bezier(0, 0.9, 1, 0.1) 2.5s;
+  animation: warning 1s linear 0s alternate infinite;
+  -webkit-animation: warning 1s linear 0s alternate infinite;
 }
 
 @keyframes warning {
-    0 % { text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);}
-    50 % { text-shadow: 0px 0px 30px rgba(255, 0, 0, 1), 0px 0px 6px rgb(129, 0, 0);}
-    100 % { text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);}
+  0 % {
+    text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);
+  }
+  50 % {
+    text-shadow: 0px 0px 30px rgba(255, 0, 0, 1), 0px 0px 6px rgb(129, 0, 0);
+  }
+  100 % {
+    text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);
+  }
 }
 
 @-webkit - keyframes warning {
-    0 % { text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);}
-    50 % { text-shadow: 0px 0px 30px rgba(255, 0, 0, 1), 0px 0px 6px rgb(129, 0, 0);}
-    100 % { text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);}
+  0 % {
+    text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);
+  }
+  50 % {
+    text-shadow: 0px 0px 30px rgba(255, 0, 0, 1), 0px 0px 6px rgb(129, 0, 0);
+  }
+  100 % {
+    text-shadow: 0px 0px 9px rgba(255, 0, 0, 0.5), 0px 0px 6px rgb(129, 0, 0);
+  }
 }
 
 .gameCanvas {
-    transition: transform 1.5s ease-out;
-    -webkit-transition: transform 1.5s ease-out;
-}
\ No newline at end of file
+  transition: transform 1.5s ease-out;
+  -webkit-transition: transform 1.5s ease-out;
+}
+/* ====== 新增加载界面样式 ====== */
+@media screen and (max-width: 600px) {
+  .loading-slide {
+    width: 100%
+  }
+}
+#startTopGallery {
+  position: absolute;
+  width: 85%;
+  height: 85%;
+  top: 10%;
+  left:7.5%;
+  perspective: 1000px;
+}
+
+.loading-slide {
+  position: absolute;
+  width: 100%;
+  height: 100%;
+  background-size: cover;
+  background-position: center;
+  opacity: 0;
+  object-fit: contain;
+  transform: translateX(100%) rotateY(30deg);
+  transition: all 1s cubic-bezier(0.4, 0, 0.2, 1);
+  border-radius: 10px;
+  box-shadow: 0 8px 20px rgba(0,0,0,0.4);
+}
+
+.loading-slide.active {
+  opacity: 1;
+  transform: translateX(0) rotateY(0);
+}